How to style an element if it contains or if it starts with another element?

I tried searching for various combinations of “style element based on content” or “style container based on child” or style “based on containing elements” but didn’t find a solution for the following two fold question:

I want to style a container #outside (or parts of it) in two cases

  • Case A: When it contains #inside
  • Case B: When it starts with #inside
#let inside(body) = {
  set text(blue)
  body
}

#let outside(body) = {
  if body.has("child") {
    set text(blue) if body.children.at(0).child == "??" // This was an attempt
  }
  "Yo, " + body
}

*How can I make this blue if it contains `#inside`?*\
#outside[listen up here′s a story #inside[I'm blue] Da ba dee da ba di.] \
#text(fill: blue, "Yo, listen up here′s a story I'm blue Da ba dee da ba di.") (Expected)

*How can I make the "Yo, " blue only if `#outside` starts with `#inside`?*\
#outside[#inside[I'm blue] Da ba dee da ba di.] \
#text(fill: blue, "Yo, I'm blue") Da ba dee da ba di. (Expected) \
#text(fill: blue, "Yo, I'm also blue") Da ba dee da ba di. (Expected)

I realize “element” may not be the best (or most specific) word both “elements” are user-defined functions.

What is the best way to achieve this? :blush:

1 Like

I think this isn’t currently achievable easily, because of this:

But GitHub - PgBiel/elembic: Framework for custom elements and types in Typst might be able to help in achieving this in the meantime.

1 Like

Thank you, that’s bad news for the case, but somewhat good news for me, as I didn’t overlook something super obvious (as many times before).

So even the #outside[#inside[ case is not detectable because, other than the content, there is nothing to check against. Am I seeing this right?

I’ve just tried:

#outside[#inside[I'm blue]<blue> Da ba dee da ba di.]

as a workaround, but the sequence doesn’t seem to contain the label either.

Here’s a hacky solution for the first question.

Basic explanation: Every outside creates a new counter. inside adds to the current outside counter. The outside styling can detect the final value of its own counter and style itself accordingly.

#show "green": set text(green)

#let oc = counter("outside")

#let inside(body) = context {
  // Why do we need -1 here? Don't know yet.
  let ic = counter("inside/" + str(oc.get().first() - 1))
  ic.step()
  set text(blue)
  body
}

// There are 2 oc counters for each `outside`.
// So that `inside` increments outside/x while inside `outside`
// and adds to outside/x+1 otherwise.
#let outside(body) = context {
  oc.step()
  let ic = counter("inside/" + str(oc.get().first()))
  let isblue = ic.final().first() > 0
  set text(blue) if isblue
  "Yo, " + body
  oc.step()
}

*How can I make this blue if it contains `#inside`?*\
#outside[listen up here′s a story #inside[I'm blue] Da ba dee da ba di.] \
#outside[listen up here′s a story I'm green Da ba dee da ba di.] \
#outside[listen up here′s a story #inside[I'm blue] Da ba dee da ba di.] \
#outside[listen up here′s a story #inside[I'm blue] Da ba dee da ba di.] \
#outside[listen up here′s a story #inside[I'm blue] Da ba dee da ba di.] \
#outside[listen up here′s a story I'm green Da ba dee da ba di.] \

#outside[Another empty sentence, should be black.]

#inside[Freely running around inside]

#outside[Empty outside.]


#text(fill: blue, "Yo, listen up here′s a story I'm blue Da ba dee da ba di.") (Expected)

*How can I make the "Yo, " blue only if `#outside` starts with `#inside`?*\
#outside[#inside[I'm blue] Da ba dee da ba di.] \
#text(fill: blue, "Yo, I'm blue") Da ba dee da ba di. (Expected) \
#text(fill: blue, "Yo, I'm also blue") Da ba dee da ba di. (Expected)

bild

I have a question about my own code… after this:

#let outside(body) = context {
  oc.step()
  let ic = counter("inside/" + str(oc.get().first()))

The first time this is used, ic gets the counter called “inside/0”, why doesn’t it get number 1? oc has been stepped just on the line before…

1 Like

Exactly.

#show <blue>: it => "Blue is attached to " + raw(repr(it), lang: "typc")

#outside[#inside[I'm blue]<blue> Da ba dee da ba di.]

image

1 Like

The .get() call is called within a context, and the context starts with counter set to 0. For .get() to see a .step(), they need to be separated by context:

#let outside(body) = {
  oc.step()
  context {
      let ic = counter("inside/" + str(oc.get().first()))

P.S. The forum is English-oriented, so writing (or leaving) 詳細 in a message is confusing unless you’re 日本語上手.

1 Like

Ah, thanks for the reminder, makes sense!

This part I don’t understand, I don’t think that can be related to anything I’ve written on the forum.

I have google translated your message, and I think I understand now.

I used a HTML <details> tag in my message. It’s localized by your web browser. I see it in Swedish, hence your Japanese (I think) was very confusing to me. So yeah, we see different things. And yeah, it’s not me who decides how it’s displayed.

bild

1 Like

See, reading the message in English and then see one small part in localized language is confusing either way (if it’s not English).

In any case, “details” is not descriptive, so you should provide the summary of what you hide.

<details><summary>the summary</summary>

```typ
stuff
```

</details>

or

[details="Summary"]
This text will be hidden
[/details]

Which is an option in the dropdown menu in the editor (to not write it manually).

Nice, very creative approach! :clap:t2:

There is currently a pull request on the way for .within so hopefully both of these cases will have non-hacky solution soon.

1 Like