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

Ranges support and begin/end members #873

Closed
wants to merge 12 commits into from
Closed

Ranges support and begin/end members #873

wants to merge 12 commits into from

Conversation

sylveon
Copy link
Contributor

@sylveon sylveon commented Mar 7, 2021

This enables WinRT collections to be used with C++20 ranges.

I added member begin/end because the existing solution (ADL-only) doesn't work with std::ranges::begin's ADL resolution - it defines a deleted template begin, and this template is a better match than C++/WinRT's ADL functions.

Fixes #807

@sylveon sylveon changed the title Base ranges support Basic ranges support Mar 7, 2021
@sylveon
Copy link
Contributor Author

sylveon commented Mar 7, 2021

Example:

auto v = winrt::single_threaded_vector<int>({1, 2, 3});
for (const auto i : v | std::views::reverse)
{
    std::cout << i;
}

@sylveon sylveon changed the title Basic ranges support Ranges support Mar 8, 2021
@sylveon sylveon changed the title Ranges support Ranges support and begin/end members Mar 8, 2021
@kennykerr kennykerr requested a review from DefaultRyan March 8, 2021 13:57
@sylveon
Copy link
Contributor Author

sylveon commented Mar 8, 2021

I'm actually thinking it might be possible to enable ranges for IIterable as well - they do require post-fix ++ but put no constraint on its return type, so it would be possible to write a post-fix ++ returning void. Thoughts?

@sylveon
Copy link
Contributor Author

sylveon commented Mar 8, 2021

With my latest commit, every IIterable can also be used with ranges:

// prints "1 3 5 7"
auto v = winrt::single_threaded_vector<int>({ 1, 2, 3, 4, 5, 6, 7 });
winrt::Windows::Foundation::Collections::IIterable<int> c(v);

for (int i : c | std::views::filter([](int i) { return i % 2 != 0; }))
{
    std::cout << i << ' ';
}

@kennykerr
Copy link
Collaborator

kennykerr commented Mar 10, 2021

Thanks for the contribution! @DefaultRyan can you take a look?

@DefaultRyan
Copy link
Member

I should be able to get to it in a day or two. I haven't been doing my homework reading up on the details of these newfangled ranges, but from what I've heard, this should be exciting.

@DefaultRyan
Copy link
Member

I'm actually thinking it might be possible to enable ranges for IIterable as well - they do require post-fix ++ but put no constraint on its return type, so it would be possible to write a post-fix ++ returning void. Thoughts?

I'm not quite convinced. Which concept is it that needs to be satisfied?

@sylveon
Copy link
Contributor Author

sylveon commented Mar 11, 2021

Which concept is it that needs to be satisfied?

std::weakly_incrementable

@sylveon
Copy link
Contributor Author

sylveon commented Mar 12, 2021

To be clear, it's std::incrementable which requires i++ to return decltype(i). std::weakly_incrementable just wants i++ to be a valid expression and it can return nothing for all it cares.

std::input_iterator only requires std::weakly_incrementable, while std::forward_iterator wants std::incrementable, so this makes IIterator<T> only comply to std::input_iterator (currently, it also only presents itself as an input iterator for the same post-fix ++ reason, nothing changes here). As explained in cppreference, algorithms over std::input_iterator should be single-pass algorithms only as well.

Making i++ return void will also emit a compiler error in case someone tries to do something like IIterator<T> last = i++, so I think it's a fair trade to enable ranges: it doesn't break any code, since post-fix ++ on IIterator<T> doesn't exist currently, and makes sure that nobody accidentally writes wrong code using this added operator.

@sylveon
Copy link
Contributor Author

sylveon commented Mar 20, 2021

Any update on this? I'd like to start using ranges in my projects but this is a pretty big blocker.

@kennykerr
Copy link
Collaborator

@DefaultRyan will hopefully get a chance to take a look soon.

I had a quick look and the obvious missing piece here is good test coverage.

@sylveon
Copy link
Contributor Author

sylveon commented Mar 22, 2021

I was thinking the same but I'm not sure how to proceed: do I create a brand new test project with C++20 or upgrade the existing one to C++20?

@kennykerr
Copy link
Collaborator

kennykerr commented Mar 22, 2021

Yes, we don't want to destabilize the existing tests. Probably best to create a new test project by copying "test", removing the existing tests, and then flipping it to C++20.

Obviously, all of the existing tests should run unchanged.

@sylveon
Copy link
Contributor Author

sylveon commented Mar 23, 2021

Just added a few unit tests, and made sure they pass locally. The IDE says some constraints aren't satisfied when editing the ranges.cpp file but it builds and runs, so it seems like some EDG bugs that should be reported.

@kennykerr
Copy link
Collaborator

Thanks Charles, I had a quick look and this is sufficiently disruptive that I'll need to run a CI build to validate the changes. Unfortunately I can only do so against branches on https://github.com/microsoft/cppwinrt - so if you don't mind please push a branch directly and reopen a PR. I have given you Write access in order to do that. I would also do a complete build and test before pushing as I did notice a few issues with some of the other test projects indicating regressions.

@kennykerr kennykerr closed this Mar 23, 2021
@sylveon
Copy link
Contributor Author

sylveon commented Mar 23, 2021

I pushed my current changes to user/sylveon/ranges, will do a full test run and fix anything that crops up this evening. I'll open a new PR once it's done :)

@sylveon sylveon mentioned this pull request Mar 24, 2021
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

No ranges support for collection types?
4 participants