Skip to content

Commit ef1f3a0

Browse files
committed
Add alert blocks in markdown
Signed-off-by: Yarden Shoham <[email protected]>
1 parent 92fda9c commit ef1f3a0

File tree

4 files changed

+106
-26
lines changed

4 files changed

+106
-26
lines changed

modules/markup/markdown/ast.go

+1-6
Original file line numberDiff line numberDiff line change
@@ -182,12 +182,7 @@ func IsColorPreview(node ast.Node) bool {
182182
return ok
183183
}
184184

185-
const (
186-
AttentionNote string = "Note"
187-
AttentionWarning string = "Warning"
188-
)
189-
190-
// Attention is an inline for a color preview
185+
// Attention is an inline for an attention
191186
type Attention struct {
192187
ast.BaseInline
193188
AttentionType string

modules/markup/markdown/goldmark.go

+64-14
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,6 @@ func (g *ASTTransformer) Transform(node *ast.Document, reader text.Reader, pc pa
5353
}
5454
}
5555

56-
attentionMarkedBlockquotes := make(container.Set[*ast.Blockquote])
5756
_ = ast.Walk(node, func(n ast.Node, entering bool) (ast.WalkStatus, error) {
5857
if !entering {
5958
return ast.WalkContinue, nil
@@ -197,18 +196,62 @@ func (g *ASTTransformer) Transform(node *ast.Document, reader text.Reader, pc pa
197196
if css.ColorHandler(strings.ToLower(string(colorContent))) {
198197
v.AppendChild(v, NewColorPreview(colorContent))
199198
}
200-
case *ast.Emphasis:
201-
// check if inside blockquote for attention, expected hierarchy is
202-
// Emphasis < Paragraph < Blockquote
203-
blockquote, isInBlockquote := n.Parent().Parent().(*ast.Blockquote)
204-
if isInBlockquote && !attentionMarkedBlockquotes.Contains(blockquote) {
205-
fullText := string(n.Text(reader.Source()))
206-
if fullText == AttentionNote || fullText == AttentionWarning {
207-
v.SetAttributeString("class", []byte("attention-"+strings.ToLower(fullText)))
208-
v.Parent().InsertBefore(v.Parent(), v, NewAttention(fullText))
209-
attentionMarkedBlockquotes.Add(blockquote)
210-
}
199+
case *ast.Blockquote:
200+
// We only want attention blockquotes when the AST looks like:
201+
// Text: "["
202+
// Text: "!TYPE"
203+
// Text(SoftLineBreak): "]"
204+
205+
// grab these nodes
206+
firstParagraph := v.FirstChild()
207+
if firstParagraph.ChildCount() < 3 {
208+
return ast.WalkContinue, nil
209+
}
210+
firstTextNode, ok := firstParagraph.FirstChild().(*ast.Text)
211+
if !ok {
212+
return ast.WalkContinue, nil
213+
}
214+
secondTextNode, ok := firstTextNode.NextSibling().(*ast.Text)
215+
if !ok {
216+
return ast.WalkContinue, nil
217+
}
218+
thirdTextNode, ok := secondTextNode.NextSibling().(*ast.Text)
219+
if !ok {
220+
return ast.WalkContinue, nil
221+
}
222+
223+
// make sure we adhere to the attention blockquote structure
224+
if string(firstTextNode.Segment.Value(reader.Source())) != "[" ||
225+
!attentionTypeRE.MatchString(string(secondTextNode.Segment.Value(reader.Source()))) ||
226+
string(thirdTextNode.Segment.Value(reader.Source())) != "]" {
227+
return ast.WalkContinue, nil
228+
}
229+
230+
// grab attention type from markdown source
231+
attentionType := strings.ToLower(strings.TrimPrefix(string(secondTextNode.Segment.Value(reader.Source())), "!"))
232+
233+
// color the blockquote
234+
v.SetAttributeString("class", []byte("gt-py-3 attention attention-"+attentionType))
235+
236+
// create an emphasis to make it bold
237+
emphasis := ast.NewEmphasis(2)
238+
emphasis.SetAttributeString("class", []byte("gt-font-bold attention-"+attentionType))
239+
firstParagraph.InsertBefore(firstParagraph, firstTextNode, emphasis)
240+
241+
// capitalize first letter
242+
attentionText := ast.NewString([]byte(strings.ToUpper(string(attentionType[0])) + string(attentionType[1:])))
243+
244+
// replace the ![TYPE] with icon+Type
245+
emphasis.AppendChild(emphasis, attentionText)
246+
for i := 0; i < 2; i++ {
247+
lineBreak := ast.NewText()
248+
lineBreak.SetSoftLineBreak(true)
249+
firstParagraph.InsertAfter(firstParagraph, emphasis, lineBreak)
211250
}
251+
firstParagraph.InsertBefore(firstParagraph, emphasis, NewAttention(attentionType))
252+
firstParagraph.RemoveChild(firstParagraph, firstTextNode)
253+
firstParagraph.RemoveChild(firstParagraph, secondTextNode)
254+
firstParagraph.RemoveChild(firstParagraph, thirdTextNode)
212255
}
213256
return ast.WalkContinue, nil
214257
})
@@ -346,10 +389,16 @@ func (r *HTMLRenderer) renderAttention(w util.BufWriter, source []byte, node ast
346389

347390
var octiconType string
348391
switch n.AttentionType {
349-
case AttentionNote:
392+
case "note":
350393
octiconType = "info"
351-
case AttentionWarning:
394+
case "tip":
395+
octiconType = "light-bulb"
396+
case "important":
397+
octiconType = "report"
398+
case "warning":
352399
octiconType = "alert"
400+
case "caution":
401+
octiconType = "stop"
353402
}
354403
_, _ = w.WriteString(string(svg.RenderHTML("octicon-" + octiconType)))
355404
} else {
@@ -418,6 +467,7 @@ func (r *HTMLRenderer) renderSummary(w util.BufWriter, source []byte, node ast.N
418467
}
419468

420469
var validNameRE = regexp.MustCompile("^[a-z ]+$")
470+
var attentionTypeRE = regexp.MustCompile("^!(NOTE|TIP|IMPORTANT|WARNING|CAUTION)$")
421471

422472
func (r *HTMLRenderer) renderIcon(w util.BufWriter, source []byte, node ast.Node, entering bool) (ast.WalkStatus, error) {
423473
if !entering {

modules/markup/sanitizer.go

+2-1
Original file line numberDiff line numberDiff line change
@@ -64,7 +64,8 @@ func createDefaultPolicy() *bluemonday.Policy {
6464
policy.AllowAttrs("class").Matching(regexp.MustCompile(`^color-preview$`)).OnElements("span")
6565

6666
// For attention
67-
policy.AllowAttrs("class").Matching(regexp.MustCompile(`^attention-\w+$`)).OnElements("strong")
67+
policy.AllowAttrs("class").Matching(regexp.MustCompile(`^gt-py-3 attention attention-\w+$`)).OnElements("blockquote")
68+
policy.AllowAttrs("class").Matching(regexp.MustCompile(`^gt-font-bold attention-\w+$`)).OnElements("strong")
6869
policy.AllowAttrs("class").Matching(regexp.MustCompile(`^attention-icon attention-\w+$`)).OnElements("span", "strong")
6970
policy.AllowAttrs("class").Matching(regexp.MustCompile(`^svg octicon-\w+$`)).OnElements("svg")
7071
policy.AllowAttrs("viewBox", "width", "height", "aria-hidden").OnElements("svg")

web_src/css/base.css

+39-5
Original file line numberDiff line numberDiff line change
@@ -1268,20 +1268,54 @@ img.ui.avatar,
12681268
border-radius: var(--border-radius);
12691269
}
12701270

1271+
.attention {
1272+
color: unset !important;
1273+
}
1274+
12711275
.attention-icon {
12721276
vertical-align: text-top;
12731277
}
12741278

1275-
.attention-note {
1276-
font-weight: unset;
1277-
color: var(--color-info-text);
1279+
blockquote.attention-note {
1280+
border-left-color: var(--color-blue-dark-2);
1281+
}
1282+
1283+
blockquote.attention-tip {
1284+
border-left-color: var(--color-success-border);
12781285
}
12791286

1280-
.attention-warning {
1281-
font-weight: unset;
1287+
blockquote.attention-important {
1288+
border-left-color: var(--color-violet-dark-2);
1289+
}
1290+
1291+
blockquote.attention-warning {
1292+
border-left-color: var(--color-warning-text);
1293+
}
1294+
1295+
blockquote.attention-caution {
1296+
border-left-color: var(--color-red-dark-2);
1297+
}
1298+
1299+
strong.attention-note, .attention-icon.attention-note {
1300+
color: var(--color-blue-dark-2);
1301+
}
1302+
1303+
strong.attention-tip, .attention-icon.attention-tip {
1304+
color: var(--color-success-border);
1305+
}
1306+
1307+
strong.attention-important, .attention-icon.attention-important {
1308+
color: var(--color-violet-dark-2);
1309+
}
1310+
1311+
strong.attention-warning, .attention-icon.attention-warning {
12821312
color: var(--color-warning-text);
12831313
}
12841314

1315+
strong.attention-caution, .attention-icon.attention-caution {
1316+
color: var(--color-red-dark-2);
1317+
}
1318+
12851319
.center:not(.popup) {
12861320
text-align: center;
12871321
}

0 commit comments

Comments
 (0)