2023-07-22 14:13:46 -04:00
|
|
|
from os import environ
|
|
|
|
from urllib.parse import urlencode
|
2024-03-22 20:29:34 -04:00
|
|
|
from datatypes import PDFDestination, PDFPage, PDFSection, PDFReference, Reference
|
2023-07-22 14:13:46 -04:00
|
|
|
from typing import assert_never
|
|
|
|
from pathlib import Path
|
|
|
|
|
2023-07-23 20:02:36 -04:00
|
|
|
|
2023-07-22 14:13:46 -04:00
|
|
|
def format_pdf_link(ref: PDFReference) -> str:
|
|
|
|
path_str = environ.get("TYPST_ROOT", None)
|
|
|
|
if path_str is None:
|
|
|
|
raise KeyError("Please set TYPST_ROOT to format links with Typst.")
|
|
|
|
typst_root = Path(path_str)
|
|
|
|
|
|
|
|
relative: bool = ref.filepath.is_relative_to(typst_root)
|
|
|
|
format_path: str
|
|
|
|
|
|
|
|
if relative:
|
|
|
|
format_path = "/" + str(ref.filepath.relative_to(typst_root))
|
|
|
|
else:
|
|
|
|
format_path = str(ref.filepath.absolute())
|
|
|
|
|
|
|
|
params = {}
|
2023-07-23 20:01:36 -04:00
|
|
|
default_label = ""
|
2023-07-22 14:13:46 -04:00
|
|
|
|
|
|
|
match ref:
|
|
|
|
case PDFPage():
|
|
|
|
params["page"] = ref.page
|
|
|
|
case PDFSection():
|
|
|
|
params["section"] = ref.title
|
2023-07-23 20:01:36 -04:00
|
|
|
default_label = ref.title
|
2024-03-22 20:29:34 -04:00
|
|
|
case PDFDestination():
|
|
|
|
params["destination"] = ref.name
|
2023-07-22 14:13:46 -04:00
|
|
|
case _ as obj:
|
|
|
|
assert_never(obj)
|
|
|
|
|
|
|
|
if relative:
|
2023-07-23 20:02:36 -04:00
|
|
|
return (
|
|
|
|
f'#lref("{format_path}?{urlencode(params)}", pdfref: true)[{default_label}]'
|
|
|
|
)
|
2023-07-22 14:13:46 -04:00
|
|
|
else:
|
2023-07-23 20:01:36 -04:00
|
|
|
return f'#link("pdfref://{format_path}?{urlencode(params)}")[{default_label}]'
|
2023-07-22 14:13:46 -04:00
|
|
|
|
2023-07-23 20:02:36 -04:00
|
|
|
|
2023-07-22 14:13:46 -04:00
|
|
|
def ref(ref: Reference) -> str:
|
|
|
|
"""Formats a Reference."""
|
|
|
|
|
|
|
|
# for now no other types are implemented
|
|
|
|
# replace this with a match/case when that happens
|
|
|
|
return format_pdf_link(ref)
|