Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature - Container Queries #16846

Open
wants to merge 21 commits into
base: master
Choose a base branch
from
Open

Feature - Container Queries #16846

wants to merge 21 commits into from

Conversation

emmauss
Copy link
Contributor

@emmauss emmauss commented Aug 28, 2024

What does the pull request do?

Container Queries allows styles to be activated for control based on the size of an ancestor, which acts as a container. This feature is similar to css's container queries.
A container can be defined by setting the ContainerType property. ContainerType is an enum with the following values.

public enum ContainerType
{
    /// <summary>
    /// The container will not be queries for any container size queries.
    /// </summary>
    Normal,

    /// <summary>
    /// The can be queried for container size queries for width.
    /// </summary>
    Width,

    /// <summary>
    /// The can be queried for container size queries for height.
    /// </summary>
    Height,

    /// <summary>
    /// The can be queried for container size queries for both width and height.
    /// </summary>
    WidthAndHeight
}

Currently only ContentControl and Border support acting as a container. To add container behavior for custom controls that do not implement these two controls, implement the Avalonia.Styling.IContainer interface.

    public interface IContainer
    {
        /// <summary>
        /// Gets or sets the <see cref="Avalonia.Layout.ContainerType"/> of the container.
        /// </summary>
        ContainerType ContainerType { get; set; }

        /// <summary>
        /// The name of the container.
        /// </summary>
        string? ContainerName { get; set; }

        /// <summary>
        /// The <see cref="VisualQueryProvider"/> of the container. This provides size information for the container query.
        /// </summary>
        VisualQueryProvider QueryProvider { get; }
    }

Queries are treated as selectors for the container, and are evaluated when the container size changes. Take the following use case

<Styles>
  <Style Selector="Border.b">
    <Setter Property="Background" Value="Red" />
    <Setter Property="Width" Value="200"/>
  </Style>
  <ContainerQuery Query="min-width:400">
    <Style Selector="Border.b">
      <Setter Property="Width" Value="400"/>
    </Style>
  </ContainerQuery>
</Styles>
<Grid ColumnDefinitions="*,200">
  <ContentControl Container="Width">
    <StackPanel>
      <Border Classes="b"/>
    </StackPanel>
  </ContentControl>
  <Button Grid.Column="1"/>
</Grid>

When ContentControl size changes, it triggers a reevaluation of any ContainerQuery attached to it. If the ContentControl's width is at least 400px, the styles in the Query are activated and the Border.b style is applied to any child that matches that selector.
Supported queries include;

  1. min-width
  2. max-width
  3. width
  4. min-height
  5. max-height
  6. height

A container query can specify which container to attach to by setting the Container property. The container will also need it's ContainerName property to be set.

<Styles>
  <Style Selector="Border.b">
    <Setter Property="Background" Value="Red" />
    <Setter Property="Width" Value="200"/>
  </Style>
  <ContainerQuery Query="min-width:400">
    <Style Selector="Border.b">
      <Setter Property="Width" Value="400"/>
    </Style>
  </ContainerQuery>

  <ContainerQuery Container="C1" ="min-width:400">
    <Style Selector="Border.b">
      <Setter Property="Width" Value="700"/>
    </Style>
  </ContainerQuery>
</Styles>
<Grid ColumnDefinitions="*,200">
  <ContentControl ContainerType="Width" ContainerName="C1">
    <StackPanel>
      <Border Classes="b"/> <!-- Should be 700px if 1st Grid column is >=400px instead of 400px -->
    </StackPanel>
  </ContentControl>
  <Button Grid.Column="1"/>
</Grid>

Only the ContainerQuery matching the C1 container name will be evaluated.

Layout

Container queries activates styles based on the size of a visual. The styles applied can change the size of the container, so the container size should remain constant during the whole layout pass.
At the beginning of the Measure pass, the size constraints of the container, i.e. the maximum size the container can be with respect to it's set Width,Height, Min and Max dimension are sent for evaluation before children are measured.
Measure proceeds as normal, with DesiredSize set according to the follow after children are measured.

  1. If a dimension(Width or Height) is queried, and the value is Infinity, the respective measured dimension is used in the DesiredSize
  2. If the dimension is not Infinity, the constraint dimension is used in the DesiredSize.

For an example, in the use case above the ContentControl queries for Width, so if the Width evaluated is PositiveInfinity, the measured width of its content will be used as the DesiredSize.Width. If it 700px, the DesiredSize.Width will be 700. The DesireSize.Height will be the height measure from the children in both cases.

This is a partial rewrite of #7938

What is the current behavior?

What is the updated/expected behavior with this PR?

How was the solution implemented (if it's not obvious)?

Checklist

Breaking changes

Obsoletions / Deprecations

Fixed issues

Fixes #16949

@avaloniaui-bot
Copy link

You can test this PR using the following package version. 11.2.999-cibuild0051541-alpha. (feed url: https://nuget-feed-all.avaloniaui.net/v3/index.json) [PRBUILDID]

@avaloniaui-bot
Copy link

You can test this PR using the following package version. 11.2.999-cibuild0051554-alpha. (feed url: https://nuget-feed-all.avaloniaui.net/v3/index.json) [PRBUILDID]

@avaloniaui-bot
Copy link

You can test this PR using the following package version. 11.2.999-cibuild0051558-alpha. (feed url: https://nuget-feed-all.avaloniaui.net/v3/index.json) [PRBUILDID]

@grokys
Copy link
Member

grokys commented Aug 29, 2024

Currently only ContentControl and Border support acting as a container.

Should probably be Decorator (base class of Border)?

@grokys
Copy link
Member

grokys commented Aug 29, 2024

The container will also need it's ContainerName property to be set.

Names are unique in a name scope. Are ContainerNames intended to be unique in any scope? Feels a little inconsistent having non-unique names. Maybe ContainerClass might be more consistent?

Thinking more about it: if that's the case then can we just use the existing Classes?

@emmauss
Copy link
Contributor Author

emmauss commented Aug 29, 2024

The container will also need it's ContainerName property to be set.

Names are unique in a name scope. Are ContainerNames intended to be unique in any scope? Feels a little inconsistent having non-unique names. Maybe ContainerClass might be more consistent?

Thinking more about it: if that's the case then can we just use the existing Classes?

Names aren't unique in css. They probably couldn't use Class as it will conflict with their existing Class concept. An element can have multiple classes, but only 1 name, but multiple elements can have the same name
We can't use existing Classes because there could be inconsistencies in matching which container is most suitable to query, especially if there are multiple container queries with different names but pointing to the same container. It would be best to make the user to specify which container to match for any query they write.

@emmauss
Copy link
Contributor Author

emmauss commented Aug 29, 2024

Currently only ContentControl and Border support acting as a container.

Should probably be Decorator (base class of Border)?

That would be the better class to implement it for.

@avaloniaui-bot
Copy link

You can test this PR using the following package version. 11.2.999-cibuild0051562-alpha. (feed url: https://nuget-feed-all.avaloniaui.net/v3/index.json) [PRBUILDID]

@emmauss emmauss force-pushed the feat-container-queries branch from 3a36f78 to 5d58ebd Compare August 30, 2024 10:40
@robloo
Copy link
Contributor

robloo commented Aug 31, 2024

I really like to see functionality expanding in this area. We had lots of related conversations with the media query branch and this brings several ideas together nicely. However, I don't think we should follow CSS this closely here. Let me explain.

Firstly we should note that this functionality (a subset) was implemented in WinUI using AdaptiveTriggers. The idea behind AdaptiveTriggers was fairly powerful -- if UWP at the time worked like WPF's Triggers this would have been nearly perfect. Avalonia doesn't have style triggers. I've originally lamented that as I think they are pretty simple to use in several cases. However, style selectors with pseudoclasses and property matching have proven pretty good for this as well covering the vast majority of use cases a trigger would have otherwise handled. So all is well on that front but we obviously can't use AdaptiveTrigger.

My instinct at this point is simply to extend the selector syntax to handle visual tree navigation. I.e. We need a syntax that can query another control's properties in the tree. We already have precedence for this: The nth-child selector works this way. The item itself doesn't have enough information to know whether the style should be applied -- but it's container parent does. Because some prior art exists here I have to think it can be generalized without a major performance impact.

Some other points:

  1. It would be great if we supported percentages (finally!)
  2. It would be great if this didn't have a string name. I realize that's like SharedSizeGroups but here it limits us. We should just have a target-type control IMO that can be any control in the tree. All controls have size/bounds/sizechanged.
  3. ContainerType doesn't make a lot of sense. You deviated quite a bit from CSS for no reason as far as I can tell. This is only for size changes anyway so basically should only have Normal and Size.

Bringing all this together, why isn't it possible to have a syntax like the following in Selectors themselves:

<Style Selector="TextBlock:parent-min-width:300">
<Style Selector="TextBlock:parent-min-width:50%">

or

<Style Selector="TextBlock:parent-min-width:300(#ParentControlNameInTree)">

These are not fully developed ideas but hopefully it gets my point across. The style selector syntax itself is certainly debatable.

@thevortexcloud
Copy link
Contributor

thevortexcloud commented Aug 31, 2024

I.e. We need a syntax that can query another control's properties in the tree.

I could be wrong/misunderstanding you, but is that not what this is?

https://docs.avaloniaui.net/docs/reference/styles/style-selector-syntax#by-property-match

@emmauss
Copy link
Contributor Author

emmauss commented Aug 31, 2024

I really like to see functionality expanding in this area. We had lots of related conversations with the media query branch and this brings several ideas together nicely. However, I don't think we should follow CSS this closely here. Let me explain.

Firstly we should note that this functionality (a subset) was implemented in WinUI using AdaptiveTriggers. The idea behind AdaptiveTriggers was fairly powerful -- if UWP at the time worked like WPF's Triggers this would have been nearly perfect. Avalonia doesn't have style triggers. I've originally lamented that as I think they are pretty simple to use in several cases. However, style selectors with pseudoclasses and property matching have proven pretty good for this as well covering the vast majority of use cases a trigger would have otherwise handled. So all is well on that front but we obviously can't use AdaptiveTrigger.

My instinct at this point is simply to extend the selector syntax to handle visual tree navigation. I.e. We need a syntax that can query another control's properties in the tree. We already have precedence for this: The nth-child selector works this way. The item itself doesn't have enough information to know whether the style should be applied -- but it's container parent does. Because some prior art exists here I have to think it can be generalized without a major performance impact.

Some other points:

1. It would be great if we supported percentages (finally!)

2. It would be great if this didn't have a string name. I realize that's like SharedSizeGroups but here it limits us. We should just have a target-type control IMO that can be any control in the tree. All controls have size/bounds/sizechanged.

3. ContainerType doesn't make a lot of sense. You deviated quite a bit from CSS for no reason as far as I can tell. This is only for size changes anyway so basically should only have Normal and Size.

Bringing all this together, why isn't it possible to have a syntax like the following in Selectors themselves:

<Style Selector="TextBlock:parent-min-width:300"> <Style Selector="TextBlock:parent-min-width:50%">

or

<Style Selector="TextBlock:parent-min-width:300(#ParentControlNameInTree)">

These are not fully developed ideas but hopefully it gets my point across. The style selector syntax itself is certainly debatable.

  1. Percentages are values for setters, not queries. Though out of scope of this PR, it is a useful feature, as only Grids have an equivalent. I don't see how it's possible to query if a control is a percentage of its available size giving that a control only knows its available size on Measure, and the size it will actually take on Arrange. By that point, it's too late to activate size styles for children.
  2. The Name is optional. If Name isn't specified, it matches the nearest container in its ancestors.
  3. Css actually has 3 types, Normal, Size and InlineSize. We have no concepts of inline in Avalonia, so the separate Width and Height types were made to account for that.

I'm not sure of how the selector idea would work in the current style system. The desired size of all controls are stored in properties that aren't AvaloniaProperties, i.e. DesireSize. It can't trigger any notification for style activators to work. To make it work would mean creating and subscribing to a size event for every control that could potentially be queried for size in a specific selector.
In a chat app, TextBlock:parent-min-width:300 will subscribe to the card border of every message, and not the scrollviewer.

@robloo
Copy link
Contributor

robloo commented Aug 31, 2024

@thevortexcloud

I could be wrong/misunderstanding you, but is that not what this is?

Well, that is a bit different because it's only checking for property matches on the control itself -- NOT another control in the tree. nth-child does actually get information from elsewhere in the tree. However, you've given me a better idea for the concept and syntax.

@emmauss

I'm not sure of how the selector idea would work in the current style system.

Yes, this a clear place where not having Width/Height properties like WPF is an issue. I'll explain more. However, it might still be possible. Also note that I'm starting at a very high level and trying to design the best API first. Then implementation will need to be considered and design revised as needed. That's just the process I'm used to. I know you are quite a bit further along than I am on this issue though.

So keeping with my main points from above:

  1. We should try to use a new selector syntax. This is cleaner -- especially in XAML -- to use. It doesn't introduce any new complexities from a user-standpoint.
  2. We already have an example of a selector getting information from elsewhere in the tree: nth-child.

Here is the new syntax proposal:

  • Selector="TextBox[^.Width=300]" : We already support > for child and '^' for parent in selectors. This is a new sytax that allows using the direct visual parent to get the property value of the Width property... Yes Width doesn't exist in Avalonia so more later
  • Selector="TextBox[#ASpecificParentContainer.Width=300]" : Now we extend the syntax to support referencing a parent control (container) by name directly. Then we read the property values off it to see if the selector matches.
  • Because Width doesn't exist in Avalonia, but it's super useful in cases like this, it could be special cased. Behind the scenes it would use the Bounds.Width value. If that is not agreeable it would be possible to add new read-only ActualWidth and ActualHeight properties to expose this information in the current system. Now doing this during measure/arrange before rendering is an issue as you pointed out. But if we special case the meaning of 'Width' and 'Height' here behind the scenes we can use what is necessary as well (DesiredSize). I'm just thinking very high level for now though...
  • Selector="TextBox[#ASpecificParentContainer.Width=lt-300]" : I'm getting more creative here (perhaps too much). But to actually make this selector useful we need less-than/greater-than support (min-width and max-width above). This is something that will be extremely useful in a lot of places though. So it could theoretically be added as a lt- or gt- prefix on the value. =lt- is a little strange so I suppose some alternate could be possible removing the equal sign.

Edit: We could actually use concepts from https://github.com/levitali/CompiledBindings?tab=readme-ov-file#xbind-markup-extension. Eventually, that powerful of syntax would be useful all over the place. Here something like Selector="TextBox[#ASpecificParentContainer.Width is gt 200 and lt 300] would be extremely powerful. Conceptually it's similar to all the work C# pattern matching is doing these days.

This API is still a stretch but it has some clear benefits:

  • It's generalized. We can use ANY parent control (by name) and ANY property value on that parent control to match. It doesn't need to be a special container. This works for any existing selectors too besides size.
  • We get less-than and greater-than support for any numerical properties
  • Seems super easy to read compared to this PR but I'm of course biased.

In a chat app, TextBlock:parent-min-width:300 will subscribe to the card border of every message, and not the scrollviewer.

That was only an example above where the default would be the first parent. Specifying a parent control by name is probably more appropriate in usage.

@emmauss
Copy link
Contributor Author

emmauss commented Aug 31, 2024

'Width' and 'Height' does exist in Avalonia. But it doesn' represent the laid out dimensions, which is the most important information for queries, of a control.

@robloo
Copy link
Contributor

robloo commented Aug 31, 2024

@emmauss As I said, I'm starting at a high level and designing the best API first. Implementation second. There are multiple options here:

  1. Special case Width and Height as I said above. These would internally route to the necessary DesiredSize or other layout dimensions you need when the selector is being matched.
  2. Use your interface ideas. Any control that implements IControlSize or something has implementation of Width and Height with the appropriate layout values. Then this selector (when it used Width or Height) would only be supported if the specified parent control implemented IControlSize (or whatever it's called). That's a hybrid design but is similar to what you are doing now.
  3. I think ActualWidth and ActualHeight properties from WPF might work here as well.

@thevortexcloud
Copy link
Contributor

thevortexcloud commented Sep 1, 2024

Well, that is a bit different because it's only checking for property matches on the control itself -- NOT another control in the tree. nth-child does actually get information from elsewhere in the tree. However, you've given me a better idea for the concept and syntax.

But can't easily combine the two? EG

TextBlock[IsEnabled=True]:nth-child(2n+3)

That is how you typically do stuff in CSS. It does lead to some very verbose queries though.

I think ActualWidth and ActualHeight properties from WPF might work here as well.

Avalonia already sort of has this. It's just hidden under the Bounds property. Width/Height can be zero but Bounds contains the actual size on screen.

@grokys
Copy link
Member

grokys commented Sep 2, 2024

Selector="TextBox[#ASpecificParentContainer.Width is gt 200 and lt 300]

I really don't like this. I've been against putting arbitrary expressions in selectors quite a few times (e.g. people have asked for selectors that can query properties on the data context). These are the reasons:

  • It's hard to read. People will start putting complex expressions in here that will make your eyes bleed
  • It's bad for performance. These expressions are going to have to be parsed at startup and then evaluated on each layout pass, and property subscriptions put in place to monitor when the properties change
  • It's going to cause layout cycles where a selector creates a circular dependency on another

We already have a way to create layouts with complex logic: code-behind. I know people have some sort of dogmatic aversion to code-behind, but code is the place for such logic, not a DSL shoved into a syntax that was never intended to support it. People complain the Rx is hard to understand, but at least it's C# - here you have the equivalent but with a custom syntax and no debugging tools.

@thevortexcloud
Copy link
Contributor

thevortexcloud commented Sep 2, 2024

t's hard to read. People will start putting complex expressions in here that will make your eyes bleed

CSS has been slowly moving in that direction. It went from being a simple text file that describes visuals to being basically a very simplified specialised programming language. People however keep taking it further with things like preprocessors that add actual programming concepts like control statements. I can't say I am a fan of it.

In the context of XAML you are better off just using some sort of calculated property at that point.

@robloo
Copy link
Contributor

robloo commented Sep 2, 2024

Please keep in mind I'm approaching this from a different viewpoint. Rather than trying to integrate a completely new concept: Container Query. I'm trying to fit the functionality into what we already have.

As noted above, the core of this functionality is related to nth-child (parent properties used in selectors) and property value matching. It's a short extension from there to use ANY arbitrary parent control properties. The complexity is trying to find a syntax that works for the less-than and greater-than constraints as well as where do we get Width/Height from considering the deviations in Avalonia's design in this area.

I really don't like this... We already have a way to create layouts with complex logic: code-behind. I know people have some sort of dogmatic aversion to code-behind, but code is the place for such logic, not a DSL shoved into a syntax that was never intended to support it.

As one who probably puts too much in code-behind and never jumped on the 100% MVVM bandwagon I agree with you more than you think. However, a few points:

  1. I could just as easily spin this around and say there are some with a dogmatic aversion to making XAML less static and more powerful ;) As a DSL we can define it to be whatever is most productive. It can certainly use upgrades in areas to extend functionality.
  2. The core of this discussion -- changing style dynamically as size changes -- is going to be done in XAML regardless of the design. So no matter what syntax we are discussing here we are discussing things that we all agree go in the XAML layer.

It's hard to read. People will start putting complex expressions in here that will make your eyes bleed

If devs want to make their own eyes bleed they ALWAYS have that ability no matter what a framework does. I would never try to design around the least capable developer and reduce the feature set for everyone else. Besides, the person that ultimately would have to deal with such issues is... the developer or company that allows that in the first place. It doesn't affect anyone else. Whenever the person responsible is also the person accountable things have a way of working themselves out.

It's bad for performance. These expressions are going to have to be parsed at startup and then evaluated on each layout pass, and property subscriptions put in place to monitor when the properties change

Well, we don't actually need full expressions (I did say I might be getting too creative). However, the subset to support less-than and greater-than is little different in cost than what is required for container queries implemented another way.

I will say though simple arithmetic with CONSTANT values (as opposed to adding properties from multiple controls) will be much less of a performance concern. If we eventually supported binding to, as a contrived example, Height of one control and then added +10 to that value with a simple arithmetic expression there would be negligible performance impact and no more circular dependency than we already have.


Bringing this back on topic: Does anyone else see my viewpoint? Trying to integrate this functionality into selectors. We are already part of the way there with how selectors currently function.

I have no commitment to any specific selector syntax. I was just throwing out some ideas. I'm sure people have better ideas for syntax that could be more acceptable. Personally, I don't really like the ContainerQuery syntax specified above. I think it's overly specific when it should be better generalized and integrated with ideas we already have.

@emmauss emmauss force-pushed the feat-container-queries branch from 5d58ebd to bb68d7b Compare September 2, 2024 15:35
/// <summary>
/// Defines the interface for a container
/// </summary>
public interface IContainer
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think this needs to be a public API at least yet


double width, height;

if (isContainer)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe that this logic should be bypassed for TopLevels and TopLevel should always implicitly be a SizeAndHeight container.

@@ -0,0 +1,36 @@
<Project Sdk="Microsoft.NET.Sdk">
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we somehow fit this demo into ControlCatalog? I'd really prefer to not have yet another set of projects in the solution.

I understand that it does require a separate TopLevel and we only have one on mobile, but we can probably just switch the MainView on a button click or something.


namespace Avalonia.Platform
{
public class VisualQueryProvider
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Shouldn't be public

/// <summary>
/// Defines a container.
/// </summary>
public class Container : StyleBase
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The name feels to be too generic and is likely to cause name clashes later. We already have IContainer that means a completely different thing.

/// <summary>
/// A query in a <see cref="Container"/>.
/// </summary>
public abstract class Query
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Again, naming feels to be too generic. Maybe StyleQuery?


namespace Avalonia.Controls
{
/// <summary>
/// Base class for controls which decorate a single child control.
/// </summary>
public class Decorator : Control
public class Decorator : Control, IContainer
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The choice to mark controls as possible containers seems to be rather arbitrary.

What was the logic behind choosing Decorator, ContentControl and ContentPresenter? Why do we have both ContentControl and ContentPresenter as containers?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Those are chosen because they are expected to have 1 child, and we can predict how it measures, thus makes more sense as containers. As for ContentControl and Presenter, that ContentPresenter should be the one container, so it is a mistake to make ControlControl one. I'll update it.
We could make all Controls be a potential container, but that won't make sense for controls that do not expect content, like Calendar and NumberUpDown

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

On second thought, even if a Control implements IContainer, it doesn't implicitly become a target for queries, as its ContainerType would be Normal. So ContentControl should be an IContainer. ContentPresenter is made to be an IContainer so allow users to make customer controls that can act as a container without implementing IContainer.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think the implementation is generic enough as-is. I see no reason a container query can't target ANY control. We can already opt-out by default with type Normal.

Keep in mind that the design of XAML is nesting of controls. That means this should be opt-out rather than opt-in. IMO the controls where it just never makes sense should have a mechanism to disable the container query provided by Control base by overriding the type.

@@ -284,6 +283,11 @@ protected override Size MeasureCore(Size availableSize)

var constraint = LayoutHelper.ApplyLayoutConstraints(this, availableSize);

if (this is Styling.IContainer container)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Note that this condition is currently always true since WindowBase is a ContentControl

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also, why is it in WindowBase and not in TopLevel?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

TopLevel doesn't have any special MeasureCore implemention, relying on the base implementation in Layoutable and the override in WindowBase

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  1. TopLevels should always act as Width+Height containers for satisfying media queries
  2. TopLevels should never be affected by the new Measure logic in Layoutable
  3. Manually setting ContainerType/ContainerName on a TopLevel is invalid and should trigger an exception

_argument = argument;
}

protected override bool EvaluateIsActive() => (CurrentContainer?.ContainerType == Layout.ContainerType.Width || CurrentContainer?.ContainerType == Layout.ContainerType.WidthAndHeight)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What happens if container type changes at runtime?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It wouldn't evaluate as long as it isn't a container type that applies to the dimension.

public enum ContainerType
{
/// <summary>
/// The container will not be queries for any container size queries.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

fix: The container will not be queried for any container size queries.
better: The container is not included in any size queries.

Normal,

/// <summary>
/// The container can be queried for container size queries for width.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

better: The container size can be queried for width.

@maxkatz6 maxkatz6 mentioned this pull request Sep 12, 2024
3 tasks
@avaloniaui-bot
Copy link

You can test this PR using the following package version. 11.3.999-cibuild0053398-alpha. (feed url: https://nuget-feed-all.avaloniaui.net/v3/index.json) [PRBUILDID]

@emmauss emmauss marked this pull request as ready for review November 19, 2024 08:35
@emmauss emmauss changed the title [RFC] Feature - Container Queries Feature - Container Queries Nov 19, 2024
@emmauss emmauss requested review from robloo and kekekeks November 19, 2024 08:35
@avaloniaui-bot
Copy link

You can test this PR using the following package version. 11.3.999-cibuild0053420-alpha. (feed url: https://nuget-feed-all.avaloniaui.net/v3/index.json) [PRBUILDID]

@avaloniaui-bot
Copy link

You can test this PR using the following package version. 11.3.999-cibuild0053555-alpha. (feed url: https://nuget-feed-all.avaloniaui.net/v3/index.json) [PRBUILDID]

@avaloniaui-bot
Copy link

You can test this PR using the following package version. 11.3.999-cibuild0054263-alpha. (feed url: https://nuget-feed-all.avaloniaui.net/v3/index.json) [PRBUILDID]

@emmauss emmauss added the needs-api-review The PR adds new public APIs that should be reviewed. label Jan 14, 2025
@MrJul
Copy link
Member

MrJul commented Jan 14, 2025

API diff for review:

namespace Avalonia.Layout
{
+    public enum ContainerType
+    {
+        Normal,
+        Width,
+        Height,
+        WidthAndHeight
+    }
}
 
namespace Avalonia.Styling
{
+    public partial class Container : StyleBase
+    {
+        public Container() { }
+        public Container(System.Func<StyleQuery?, StyleQuery> query, System.String? containerName = null) { }
+        public System.String? ContainerName { get { throw null; } set { } }
+        public StyleQuery? Query { get { throw null; } set { } }
+        public override System.String ToString() { throw null; }
+    }
+
+    public abstract partial class ContainerQuery<T> : StyleQuery
+    {
+        public ContainerQuery(StyleQuery? previous, T argument) { }
+        protected T Argument { get { throw null; } }
+        public override System.String ToString(Container? owner) { throw null; }
+    }
+

namespace Avalonia.Styling
{
+    public enum QueryComparisonOperator
+    {
+        None,
+        Equals,
+        LessThan,
+        GreaterThan,
+        LessThanOrEquals,
+        GreaterThanOrEquals,
+    }
+

+    public static partial class StyleQueries
+    {
+        public static StyleQuery And(params StyleQuery[] queries) { throw null; }
+        public static StyleQuery And(System.Collections.Generic.IReadOnlyList<StyleQuery> query) { throw null; }
+        public static StyleQuery Height(this StyleQuery? previous, QueryComparisonOperator @operator, System.Double value) { throw null; }
+        public static StyleQuery Or(params StyleQuery[] queries) { throw null; }
+        public static StyleQuery Or(System.Collections.Generic.IReadOnlyList<StyleQuery> query) { throw null; }
+        public static StyleQuery Width(this StyleQuery? previous, QueryComparisonOperator @operator, System.Double value) { throw null; }
+    }
+
+    public abstract partial class StyleQuery
+    {
+        protected StyleQuery() { }
+        public override System.String ToString() { throw null; }
+        public abstract System.String ToString(Container? owner);
+    }
+

namespace Avalonia.Utilities
{
     public static partial class IdentifierParser
     {
+        public static System.ReadOnlySpan<System.Char> ParseNumber(this ref CharacterReader r) { throw null; }
     }
}

namespace Avalonia.Controls
{
     public partial class ContentControl : Primitives.TemplatedControl
     {
+        public static readonly Avalonia.StyledProperty<System.String?> ContainerNameProperty;
+        public static readonly Avalonia.StyledProperty<Avalonia.Layout.ContainerType> ContainerTypeProperty;
+        public System.String? ContainerName { get { throw null; } set { } }
+        public Avalonia.Layout.ContainerType? ContainerType { get { throw null; } set { } }
+        public System.Object? Content { get { throw null; } set { } 
     }
}

namespace Avalonia.Controls
{
     public partial class Decorator : Control
     {
+        public static readonly Avalonia.StyledProperty<System.String?> ContainerNameProperty;
+        public static readonly Avalonia.StyledProperty<Avalonia.Layout.ContainerType> ContainerTypeProperty;
+        public System.String? ContainerName { get { throw null; } set { } }
+        public Avalonia.Layout.ContainerType? ContainerType { get { throw null; } set { } }
+        public static readonly Avalonia.StyledProperty<System.String?> ContainerNameProperty;
+        public static readonly Avalonia.StyledProperty<Avalonia.Layout.ContainerType> ContainerTypeProperty;
     }
}

namespace Avalonia.Controls.Presenters
{
     public partial class ContentPresenter : Control, IContainer
     { 
+        public System.String? ContainerName { get { throw null; } set { } }
+        public Avalonia.Layout.ContainerType ContainerType { get { throw null; } set { } }
     }
}

@MrJul
Copy link
Member

MrJul commented Jan 21, 2025

Notes from the API review meeting:

The API needs changes.

ContainerType is confusing, rename to ContainerSizing or a better name.
Container should be renamed to ContainerQuery.
ContainerName is a bit confusing (what's the difference with Name for a newcomer?).

The various properties should probably be attached properties instead (possible containing class name: ContainerSizing, ContainerLayout?) to avoid "polluting" the public API of all standard controls.

IdentifierParser: make ParseNumber internal. IdentifierParser should be internal for v12.

@MrJul MrJul added api-change-requested The new public APIs need some changes. and removed needs-api-review The PR adds new public APIs that should be reviewed. labels Jan 21, 2025
@avaloniaui-bot
Copy link

You can test this PR using the following package version. 11.3.999-cibuild0054473-alpha. (feed url: https://nuget-feed-all.avaloniaui.net/v3/index.json) [PRBUILDID]

@avaloniaui-bot
Copy link

You can test this PR using the following package version. 11.3.999-cibuild0054485-alpha. (feed url: https://nuget-feed-all.avaloniaui.net/v3/index.json) [PRBUILDID]

@emmauss
Copy link
Contributor Author

emmauss commented Jan 23, 2025

Api Updates.
Dropped IContainer.
IContainer Properties are now attached properties on Layoutable, using new Container class as owner.
Style type Container -> ContainerQuery.
ContainerType property -> Container.Sizing with enum ContainerSizing
ContainerName property -> Container.Name
ContainerQuery.ContainerName -> ContainerQuery.Name
IdentifierParser.ParseNumber is now internal

@avaloniaui-bot
Copy link

You can test this PR using the following package version. 11.3.999-cibuild0054491-alpha. (feed url: https://nuget-feed-all.avaloniaui.net/v3/index.json) [PRBUILDID]

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
api-change-requested The new public APIs need some changes. feature
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Container Queries
9 participants