Skip to content

Commit

Permalink
[asciidoc] basic toc handling
Browse files Browse the repository at this point in the history
  • Loading branch information
rmannibucau committed Dec 30, 2023
1 parent 808d59c commit e8ed220
Show file tree
Hide file tree
Showing 5 changed files with 233 additions and 13 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -109,6 +109,10 @@ private Path assetsDir(final Configuration configuration, final String attribute

@Override
public void visitBody(final Body body) {
if (!"none".equals(attr("toc", "toc", "none", state.document.header().attributes()))) {
visitToc(body);
}

state.stackChain(body.children(), () -> Visitor.super.visitBody(body));
}

Expand Down Expand Up @@ -318,17 +322,22 @@ public void visitHeader(final Header header) {
@Override
public void visitSection(final Section element) {
state.stackChain(element.children(), () -> {
final var titleRenderer = new AsciidoctorLikeHtmlRenderer(configuration);
titleRenderer.visitElement(element.title() instanceof Text t && t.options().isEmpty() && t.style().isEmpty() ?
new Text(t.style(), t.value(), Map.of("nowrap", "")) :
element);
final var title = titleRenderer.result();

builder.append(" <div");
writeCommonAttributes(element.options(), c -> "sect" + (element.level() - 1) + (c == null ? "" : (' ' + c)));
if (!element.options().containsKey("id")) {
builder.append(" id=\"").append(IdGenerator.forTitle(title)).append("\"");
}
builder.append(">\n");
builder.append(" <h").append(element.level());
writeCommonAttributes(element.options(), null);
builder.append(">");
final var titleRenderer = new AsciidoctorLikeHtmlRenderer(configuration);
titleRenderer.visitElement(element.title() instanceof Text t && t.options().isEmpty() && t.style().isEmpty() ?
new Text(t.style(), t.value(), Map.of("nowrap", "")) :
element);
builder.append(titleRenderer.result());
builder.append(title);
builder.append("</h").append(element.level()).append(">\n");
builder.append(" <div class=\"sectionbody\">\n");
Visitor.super.visitSection(element);
Expand Down Expand Up @@ -748,6 +757,22 @@ public void visitMacro(final Macro element) {
}
}

protected void visitToc(final Body body) {
final int toclevels = Integer.parseInt(attr("toclevels", "toclevels", "2", state.document.header().attributes()));
if (toclevels < 1) {
return;
}

builder.append(" <div id=\"toc\" class=\"").append(attr("toc-class", "toc-class", "toc", state.document.header().attributes())).append("\">\n");
if (state.document.header().title() != null && !state.document.header().title().isBlank()) {
builder.append(" <div id=\"toctitle\">").append(state.document.header().title()).append("</div>\n");
}
final var toc = new TocVisitor(toclevels, 1);
toc.visitBody(body);
builder.append(toc.result());
builder.append(" </div>\n");
}

// todo: enhance
protected void visitXref(final Macro element) {
var target = element.label();
Expand Down Expand Up @@ -1026,7 +1051,9 @@ public Configuration setAttributes(final Map<String, String> attributes) {
}

private static class State implements AutoCloseable {
private Document document;
private static final Document EMPTY_DOC = new Document(new Header("", null, null, Map.of()), new Body(List.of()));

private Document document = EMPTY_DOC;
private List<Element> currentChain = null;
private boolean hasStem = false;
private boolean sawPreamble = false;
Expand All @@ -1035,7 +1062,7 @@ private static class State implements AutoCloseable {

@Override
public void close() {
document = null;
document = EMPTY_DOC;
currentChain = null;
sawPreamble = false;
inCallOut = false;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
/*
* Copyright (c) 2020 - 2023 - Yupiik SAS - https://www.yupiik.com
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package io.yupiik.asciidoc.renderer.html;

import java.util.regex.Pattern;

import static java.util.Locale.ROOT;

public final class IdGenerator {
private static final Pattern TAGS = Pattern.compile("<[^>]+>");
private static final Pattern FORBIDDEN_CHARS = Pattern.compile("[^\\w]+");

private IdGenerator() {
// no-op
}

public static String forTitle(final String title) {
return "_" + FORBIDDEN_CHARS.matcher(TAGS.matcher(title).replaceAll("").toLowerCase(ROOT)
.replace(" ", "_")
.replace("\n", ""))
.replaceAll("");
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,92 @@
/*
* Copyright (c) 2020 - 2023 - Yupiik SAS - https://www.yupiik.com
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package io.yupiik.asciidoc.renderer.html;

import io.yupiik.asciidoc.model.Body;
import io.yupiik.asciidoc.model.Element;
import io.yupiik.asciidoc.model.Section;
import io.yupiik.asciidoc.model.Text;
import io.yupiik.asciidoc.renderer.Visitor;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Map;

import static java.util.Locale.ROOT;
import static java.util.Optional.ofNullable;
import static java.util.stream.Collectors.joining;

public class TocVisitor implements Visitor<StringBuilder> {
private final int maxLevel;
private final int currentLevel;
private final Collection<Section> sections = new ArrayList<>();

public TocVisitor(final int toclevels, final int currentLevel) {
this.maxLevel = toclevels;
this.currentLevel = currentLevel;
}

@Override
public void visitSection(final Section element) {
if (element.level() == currentLevel) {
sections.add(element);
}
}

@Override
public StringBuilder result() {
final var builder = new StringBuilder();
if (sections.isEmpty()) {
return builder;
}

builder.append(" <ul class=\"sectlevel").append(currentLevel).append("\">\n");
if (currentLevel == maxLevel) {
builder.append(sections.stream()
.map(it -> {
final var title = title(it.title());
return " <li><a href=\"#" + id(it, title) + "\">" + title + "</a></li>";
})
.collect(joining("\n", "", "\n")));
} else {
builder.append(sections.stream()
.map(it -> {
final var tocVisitor = new TocVisitor(maxLevel, currentLevel + 1);
tocVisitor.visitBody(new Body(it.children()));
final var children = tocVisitor.result().toString();
final var title = title(it.title());
return " <li><a href=\"#" + id(it, title) + "\">" + title + "</a>\n" + children + " </li>";
})
.collect(joining("\n", "", "\n")));
}
builder.append(" </ul>\n");
return builder;
}

private String id(final Section section, final String title) {
return ofNullable(section.options().get("id"))
// todo: better sanitization
.orElseGet(() -> IdGenerator.forTitle(title));
}

private String title(final Element title) {
final var titleRenderer = new AsciidoctorLikeHtmlRenderer(new AsciidoctorLikeHtmlRenderer.Configuration());
titleRenderer.visitElement(title instanceof Text t && t.options().isEmpty() && t.style().isEmpty() ?
new Text(t.style(), t.value(), Map.of("nowrap", "")) :
title);
return titleRenderer.result();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,7 @@ void metaInPreamble() {
<blockquote>
Blabla. </blockquote>
</div> </div>
<div class="sect1">
<div class="sect1" id="_whats_new">
<h2>What's new?</h2>
<div class="sectionbody">
<div class="ulist">
Expand Down Expand Up @@ -124,7 +124,7 @@ public record Foo() {}
</p>
</div>
</div>
<div class="sect1">
<div class="sect1" id="_second_part">
<h2>Second part</h2>
<div class="sectionbody">
<div class="paragraph">
Expand Down Expand Up @@ -380,7 +380,7 @@ void stem() {
And inline stem:[[[a,b\\],[c,d\\]\\]((n),(k))] too.
""", """
<div class="sect0">
<div class="sect0" id="_some_formulas">
<h1>Some formulas</h1>
<div class="sectionbody">
<div class="stemblock">
Expand Down Expand Up @@ -415,7 +415,7 @@ void embeddedImage(@TempDir final Path work) throws IOException {
.setAttributes(Map.of("noheader", "true", "data-uri", "")));
renderer.visitBody(doc);
assertEquals("""
<div class="sect0">
<div class="sect0" id="_test">
<h1>Test</h1>
<div class="sectionbody">
<div class="imageblock">
Expand All @@ -442,7 +442,7 @@ public record UserId(String name) {}
.setAttributes(Map.of("noheader", "true", "data-uri", "false"/*true would mean we depend on the http url at test time, we don't want that*/)));
renderer.visitBody(doc);
assertEquals("""
<div class="sect0">
<div class="sect0" id="_test">
<h1>Test</h1>
<div class="sectionbody">
Expand Down Expand Up @@ -474,7 +474,7 @@ void ascii2svg() {
'-------------------------'
....
""", """
<div class="sect0">
<div class="sect0" id="_test">
<h1>Test</h1>
<div class="sectionbody">
<img src="" alt="a2s">
Expand Down

0 comments on commit e8ed220

Please sign in to comment.