mirror of
https://github.com/Cysharp/UniTask.git
synced 2026-05-15 19:40:09 +00:00
Compare commits
129 Commits
2.0.3-prev
...
fix-il2cpp
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
37cd00d347 | ||
|
|
859c4d706f | ||
|
|
7289fe6e25 | ||
|
|
0c33977f5a | ||
|
|
4d4466e801 | ||
|
|
79330d7cdb | ||
|
|
680ce1098b | ||
|
|
2337d705ec | ||
|
|
d2880a818f | ||
|
|
86ea128bf4 | ||
|
|
a66f378622 | ||
|
|
265f88584b | ||
|
|
686394c861 | ||
|
|
8bad158ab4 | ||
|
|
5e59e7ec86 | ||
|
|
8bb0a48720 | ||
|
|
b4468b4eba | ||
|
|
0725bd1b30 | ||
|
|
ebd80243e0 | ||
|
|
f1ac469058 | ||
|
|
2e0b603d25 | ||
|
|
1d90a40f66 | ||
|
|
1bec3f507e | ||
|
|
1d5ecbb3ab | ||
|
|
e1a4aeb9da | ||
|
|
43f1bb4d85 | ||
|
|
8b7a0e9b15 | ||
|
|
da329e19d1 | ||
|
|
b8260d4e91 | ||
|
|
345e32aaf0 | ||
|
|
d7bef8c5b5 | ||
|
|
b2f82df4d3 | ||
|
|
83596b3d1f | ||
|
|
0022598a1c | ||
|
|
f4294d3752 | ||
|
|
a1444c0b39 | ||
|
|
e1d5359d73 | ||
|
|
239bf749b6 | ||
|
|
2bf3b1e172 | ||
|
|
d225de201f | ||
|
|
c3b8a3852d | ||
|
|
c31dab888e | ||
|
|
d4cf59bd2f | ||
|
|
d5edc3acd3 | ||
|
|
130286e8c2 | ||
|
|
a9baa52309 | ||
|
|
3001996298 | ||
|
|
bfcd18aabb | ||
|
|
96aa299e7e | ||
|
|
24faa34418 | ||
|
|
21bf08a6b3 | ||
|
|
d5db96b913 | ||
|
|
a8455af16d | ||
|
|
2290b14532 | ||
|
|
90c5a6311b | ||
|
|
6e0ad3623b | ||
|
|
005e02a1fa | ||
|
|
10fb8060fa | ||
|
|
35b933730b | ||
|
|
7ab9467069 | ||
|
|
598312ba61 | ||
|
|
985aa5c43a | ||
|
|
10eff95a42 | ||
|
|
d27d6d5d9d | ||
|
|
b8c109848e | ||
|
|
8b7f832c0f | ||
|
|
7cce0f48e5 | ||
|
|
8a56838111 | ||
|
|
ff15e00003 | ||
|
|
f60d2c51fb | ||
|
|
6dfb969015 | ||
|
|
da7e9fc4b3 | ||
|
|
70385c4115 | ||
|
|
51ba740413 | ||
|
|
f3e3ba8864 | ||
|
|
07cf65c1ec | ||
|
|
eca5b1c096 | ||
|
|
c74ce14ad1 | ||
|
|
f59c56506f | ||
|
|
896eef1ee4 | ||
|
|
ec0123eec7 | ||
|
|
78f56b9b33 | ||
|
|
1d88ed85bc | ||
|
|
2b7986da19 | ||
|
|
c3d22968e1 | ||
|
|
0e25122ee2 | ||
|
|
4504d84aa8 | ||
|
|
2b87cadba3 | ||
|
|
21dc83c641 | ||
|
|
3b593f349c | ||
|
|
962c215e3b | ||
|
|
42dcfdbcdc | ||
|
|
6d7e6ec871 | ||
|
|
36d53a3bcb | ||
|
|
ea9e61c2e1 | ||
|
|
a52c26102b | ||
|
|
e31c87b8a8 | ||
|
|
cc165a6897 | ||
|
|
f99910d802 | ||
|
|
997b0b3710 | ||
|
|
ec7064083a | ||
|
|
07cccfddd6 | ||
|
|
f07527cd06 | ||
|
|
7b273c4bd1 | ||
|
|
d36e7987b3 | ||
|
|
bbd5686816 | ||
|
|
fb1152d8f4 | ||
|
|
7a306118f5 | ||
|
|
efaf3ee8f5 | ||
|
|
2e4fe90956 | ||
|
|
e33d572104 | ||
|
|
2b2af9e455 | ||
|
|
d003597662 | ||
|
|
ec0a8f5a8b | ||
|
|
49ba57f20a | ||
|
|
6f4d1183cc | ||
|
|
dd18c9fff8 | ||
|
|
21f5f78ff1 | ||
|
|
1729f389db | ||
|
|
6d37bb7bac | ||
|
|
957adfad7a | ||
|
|
3ef889e17d | ||
|
|
c73af7390f | ||
|
|
c5b4376486 | ||
|
|
ba65049dd8 | ||
|
|
ee58aab0a9 | ||
|
|
859eaa2278 | ||
|
|
79f770e687 | ||
|
|
8ff4de67a1 |
@@ -1,114 +0,0 @@
|
|||||||
version: 2.1
|
|
||||||
executors:
|
|
||||||
unity:
|
|
||||||
# https://hub.docker.com/r/gableroux/unity3d/tags
|
|
||||||
parameters:
|
|
||||||
version: {type: string}
|
|
||||||
docker:
|
|
||||||
- image: gableroux/unity3d:<< parameters.version >>
|
|
||||||
go:
|
|
||||||
docker:
|
|
||||||
- image: circleci/golang
|
|
||||||
commands:
|
|
||||||
unity_activate:
|
|
||||||
parameters:
|
|
||||||
unity_version: {type: string}
|
|
||||||
unity_license: {type: string}
|
|
||||||
steps:
|
|
||||||
# get activation file, if fail to activate unity, use this key and activate from https://license.unity3d.com/manual
|
|
||||||
- run: apt update && apt install libunwind8 -y
|
|
||||||
- run: /opt/Unity/Editor/Unity -quit -batchmode -nographics -logFile -createManualActivationFile || exit 0
|
|
||||||
- run: cat Unity_v<< parameters.unity_version >>.alf
|
|
||||||
# get from UNITY_LICENSE envvar(base64 encoded(cat foo.ulf | base64 )), this file is generated from above manual activation
|
|
||||||
- run: echo << parameters.unity_license >> | base64 -di >> .circleci/Unity.ulf
|
|
||||||
- run: /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -manualLicenseFile .circleci/Unity.ulf || exit 0
|
|
||||||
jobs:
|
|
||||||
build-and-test:
|
|
||||||
parameters:
|
|
||||||
unity_version: {type: string}
|
|
||||||
unity_license: {type: string}
|
|
||||||
executor:
|
|
||||||
name: unity
|
|
||||||
version: << parameters.unity_version >>
|
|
||||||
steps:
|
|
||||||
- checkout
|
|
||||||
- unity_activate:
|
|
||||||
unity_version: << parameters.unity_version >>
|
|
||||||
unity_license: << parameters.unity_license >>
|
|
||||||
- run:
|
|
||||||
name: Build Linux(Mono)
|
|
||||||
command: /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -projectPath . -executeMethod UnitTestBuilder.BuildUnitTest /headless /ScriptBackend Mono2x /BuildTarget StandaloneLinux64
|
|
||||||
working_directory: .
|
|
||||||
# TODO:check unity version and packages...
|
|
||||||
# - run: ./bin/UnitTest/StandaloneLinux64_Mono2x/test
|
|
||||||
build-and-create-package:
|
|
||||||
parameters:
|
|
||||||
unity_version: {type: string}
|
|
||||||
unity_license: {type: string}
|
|
||||||
executor:
|
|
||||||
name: unity
|
|
||||||
version: << parameters.unity_version >>
|
|
||||||
steps:
|
|
||||||
- checkout
|
|
||||||
- unity_activate:
|
|
||||||
unity_version: << parameters.unity_version >>
|
|
||||||
unity_license: << parameters.unity_license >>
|
|
||||||
- run:
|
|
||||||
name: Export unitypackage
|
|
||||||
command: /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -projectPath . -executeMethod PackageExporter.Export
|
|
||||||
working_directory: .
|
|
||||||
- store_artifacts:
|
|
||||||
path: ./UniRx.Async.unitypackage
|
|
||||||
destination: /UniRx.Async.unitypackage
|
|
||||||
# upload to github by ghr
|
|
||||||
upload-github:
|
|
||||||
executor: go
|
|
||||||
steps:
|
|
||||||
- attach_workspace:
|
|
||||||
at: .
|
|
||||||
- run: go get github.com/tcnksm/ghr
|
|
||||||
- run: ghr -t ${GITHUB_TOKEN} -u ${CIRCLE_PROJECT_USERNAME} -r ${CIRCLE_PROJECT_REPONAME} ${CIRCLE_TAG} .
|
|
||||||
- store_artifacts:
|
|
||||||
path: UniRx.Async.unitypackage
|
|
||||||
destination: UniRx.Async.unitypackage
|
|
||||||
workflows:
|
|
||||||
version: 2
|
|
||||||
build-unity:
|
|
||||||
jobs:
|
|
||||||
# does not exists yet.
|
|
||||||
# - build-and-test:
|
|
||||||
# unity_version: 2019.3.0a2
|
|
||||||
# unity_license: ${UNITY_LICENSE_2019_3}
|
|
||||||
# - build-and-test:
|
|
||||||
# unity_version: 2019.2.0b2
|
|
||||||
# unity_license: ${UNITY_LICENSE_2019_2}
|
|
||||||
- build-and-test:
|
|
||||||
unity_version: 2019.1.2f1
|
|
||||||
unity_license: ${UNITY_LICENSE_2019_1}
|
|
||||||
filters:
|
|
||||||
tags:
|
|
||||||
only: /.*/
|
|
||||||
# test asmdef will not found.
|
|
||||||
# - build-and-test:
|
|
||||||
# unity_version: 2018.4.0f1
|
|
||||||
# unity_license: ${UNITY_LICENSE_2018_4}
|
|
||||||
# # UniTask minimum support version is 2018.3(C# 7.x)
|
|
||||||
# - build-and-test:
|
|
||||||
# unity_version: 2018.3.12f1
|
|
||||||
# unity_license: ${UNITY_LICENSE_2018_3}
|
|
||||||
- build-and-create-package:
|
|
||||||
unity_version: 2019.1.2f1
|
|
||||||
unity_license: ${UNITY_LICENSE_2019_1}
|
|
||||||
filters:
|
|
||||||
tags:
|
|
||||||
only: /^\d\.\d\.\d.*/
|
|
||||||
branches:
|
|
||||||
ignore: /.*/
|
|
||||||
- upload-github:
|
|
||||||
requires:
|
|
||||||
- build-and-create-package
|
|
||||||
filters:
|
|
||||||
tags:
|
|
||||||
only: /^\d\.\d\.\d.*/
|
|
||||||
branches:
|
|
||||||
ignore: /.*/
|
|
||||||
73
.github/workflows/build-debug.yml
vendored
Normal file
73
.github/workflows/build-debug.yml
vendored
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
name: Build-Debug
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- "**"
|
||||||
|
tags:
|
||||||
|
- "!*" # not a tag push
|
||||||
|
pull_request:
|
||||||
|
types:
|
||||||
|
- opened
|
||||||
|
- synchronize
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build-dotnet:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
env:
|
||||||
|
DOTNET_CLI_TELEMETRY_OPTOUT: 1
|
||||||
|
DOTNET_SKIP_FIRST_TIME_EXPERIENCE: 1
|
||||||
|
NUGET_XMLDOC_MODE: skip
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: actions/setup-dotnet@v1
|
||||||
|
with:
|
||||||
|
dotnet-version: 3.1.101
|
||||||
|
- run: dotnet test -c Debug ./src/UniTask.NetCoreTests/UniTask.NetCoreTests.csproj
|
||||||
|
|
||||||
|
build-unity:
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
unity: ['2019.3.9f1', '2020.1.0b5']
|
||||||
|
include:
|
||||||
|
- unity: 2019.3.9f1
|
||||||
|
license: UNITY_2019_3
|
||||||
|
- unity: 2020.1.0b5
|
||||||
|
license: UNITY_2020_1
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
container:
|
||||||
|
# with linux-il2cpp. image from https://hub.docker.com/r/gableroux/unity3d/tags
|
||||||
|
image: gableroux/unity3d:${{ matrix.unity }}-linux-il2cpp
|
||||||
|
steps:
|
||||||
|
- run: apt update && apt install git -y
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
# create unity activation file and store to artifacts.
|
||||||
|
- run: /opt/Unity/Editor/Unity -quit -batchmode -nographics -logFile -createManualActivationFile || exit 0
|
||||||
|
- uses: actions/upload-artifact@v1
|
||||||
|
with:
|
||||||
|
name: Unity_v${{ matrix.unity }}.alf
|
||||||
|
path: ./Unity_v${{ matrix.unity }}.alf
|
||||||
|
# activate Unity from manual license file(ulf)
|
||||||
|
- run: echo -n "$UNITY_LICENSE" >> .Unity.ulf
|
||||||
|
env:
|
||||||
|
UNITY_LICENSE: ${{ secrets[matrix.license] }}
|
||||||
|
- name: Activate Unity, always returns a success. But if a subsequent run fails, the activation may have failed(if succeeded, shows `Next license update check is after` and not shows other message(like GUID != GUID). If fails not). In that case, upload the artifact's .alf file to https://license.unity3d.com/manual to get the .ulf file and set it to secrets.
|
||||||
|
run: /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -manualLicenseFile .Unity.ulf || exit 0
|
||||||
|
|
||||||
|
# Execute scripts: RuntimeUnitTestToolkit
|
||||||
|
- name: Build UnitTest(Linux64, mono)
|
||||||
|
run: /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -projectPath . -executeMethod UnitTestBuilder.BuildUnitTest /headless /ScriptBackend mono /BuildTarget StandaloneLinux64
|
||||||
|
working-directory: src/UniTask
|
||||||
|
- name: Execute UnitTest
|
||||||
|
run: ./src/UniTask/bin/UnitTest/StandaloneLinux64_Mono2x/test
|
||||||
|
|
||||||
|
# Execute scripts: Export Package
|
||||||
|
- name: Export unitypackage
|
||||||
|
run: /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -projectPath . -executeMethod PackageExporter.Export
|
||||||
|
working-directory: src/UniTask
|
||||||
|
|
||||||
|
# Store artifacts.
|
||||||
|
- uses: actions/upload-artifact@v2
|
||||||
|
with:
|
||||||
|
name: UniTask.unitypackage.zip
|
||||||
|
path: ./src/UniTask/*.unitypackage
|
||||||
31
.github/workflows/build-docs.yml
vendored
Normal file
31
.github/workflows/build-docs.yml
vendored
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
name: build-docs
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
- feature/docs
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
run-docfx:
|
||||||
|
if: "!(contains(github.event.head_commit.message, '[skip ci]') || contains(github.event.head_commit.message, '[ci skip]'))"
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
repository: Cysharp/DocfxTemplate
|
||||||
|
path: docs/_DocfxTemplate
|
||||||
|
- uses: Kirbyrawr/docfx-action@master
|
||||||
|
name: Docfx metadata
|
||||||
|
with:
|
||||||
|
args: metadata docs/docfx.json
|
||||||
|
- uses: Kirbyrawr/docfx-action@master
|
||||||
|
name: Docfx build
|
||||||
|
with:
|
||||||
|
args: build docs/docfx.json
|
||||||
|
- name: Publish to GitHub Pages
|
||||||
|
uses: peaceiris/actions-gh-pages@v3
|
||||||
|
with:
|
||||||
|
github_token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
publish_dir: docs/_site
|
||||||
105
.github/workflows/build-release.yml
vendored
Normal file
105
.github/workflows/build-release.yml
vendored
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
name: Build-Release
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
tags:
|
||||||
|
- "[0-9]+.[0-9]+.[0-9]+*"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build-dotnet:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
env:
|
||||||
|
DOTNET_CLI_TELEMETRY_OPTOUT: 1
|
||||||
|
DOTNET_SKIP_FIRST_TIME_EXPERIENCE: 1
|
||||||
|
NUGET_XMLDOC_MODE: skip
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: actions/setup-dotnet@v1
|
||||||
|
with:
|
||||||
|
dotnet-version: 3.1.101
|
||||||
|
# set release tag(*.*.*) to env.GIT_TAG
|
||||||
|
- run: echo ::set-env name=GIT_TAG::${GITHUB_REF#refs/tags/}
|
||||||
|
|
||||||
|
# build and pack
|
||||||
|
- run: dotnet build -c Release -p:Version=${{ env.GIT_TAG }}
|
||||||
|
- run: dotnet test -c Release --no-build
|
||||||
|
- run: dotnet pack ./src/UniTask.NetCore/UniTask.NetCore.csproj -c Release --no-build -p:Version=${{ env.GIT_TAG }}
|
||||||
|
|
||||||
|
# Store artifacts.
|
||||||
|
- uses: actions/upload-artifact@v1
|
||||||
|
with:
|
||||||
|
name: nuget
|
||||||
|
path: ./src/UniTask.NetCore/bin/Release/UniTask.${{ env.GIT_TAG }}.nupkg
|
||||||
|
|
||||||
|
build-unity:
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
unity: ['2019.3.9f1']
|
||||||
|
include:
|
||||||
|
- unity: 2019.3.9f1
|
||||||
|
license: UNITY_2019_3
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
container:
|
||||||
|
# with linux-il2cpp. image from https://hub.docker.com/r/gableroux/unity3d/tags
|
||||||
|
image: gableroux/unity3d:${{ matrix.unity }}-linux-il2cpp
|
||||||
|
steps:
|
||||||
|
- run: apt update && apt install git -y
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- run: echo -n "$UNITY_LICENSE" >> .Unity.ulf
|
||||||
|
env:
|
||||||
|
UNITY_LICENSE: ${{ secrets[matrix.license] }}
|
||||||
|
- run: /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -manualLicenseFile .Unity.ulf || exit 0
|
||||||
|
|
||||||
|
# set release tag(*.*.*) to env.GIT_TAG
|
||||||
|
- run: echo ::set-env name=GIT_TAG::${GITHUB_REF#refs/tags/}
|
||||||
|
|
||||||
|
# Execute scripts: Export Package
|
||||||
|
- name: Export unitypackage
|
||||||
|
run: /opt/Unity/Editor/Unity -quit -batchmode -nographics -silent-crashes -logFile -projectPath . -executeMethod PackageExporter.Export
|
||||||
|
working-directory: src/UniTask
|
||||||
|
|
||||||
|
# Store artifacts.
|
||||||
|
- uses: actions/upload-artifact@v2
|
||||||
|
with:
|
||||||
|
name: UniTask.${{ env.GIT_TAG }}.unitypackage
|
||||||
|
path: ./src/UniTask/UniTask.${{ env.GIT_TAG }}.unitypackage
|
||||||
|
|
||||||
|
create-release:
|
||||||
|
needs: [build-dotnet, build-unity]
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
env:
|
||||||
|
DOTNET_CLI_TELEMETRY_OPTOUT: 1
|
||||||
|
DOTNET_SKIP_FIRST_TIME_EXPERIENCE: 1
|
||||||
|
NUGET_XMLDOC_MODE: skip
|
||||||
|
steps:
|
||||||
|
# setup dotnet for nuget push
|
||||||
|
- uses: actions/setup-dotnet@v1
|
||||||
|
with:
|
||||||
|
dotnet-version: 3.1.101
|
||||||
|
# set release tag(*.*.*) to env.GIT_TAG
|
||||||
|
- run: echo ::set-env name=GIT_TAG::${GITHUB_REF#refs/tags/}
|
||||||
|
|
||||||
|
# Create Releases
|
||||||
|
- uses: actions/create-release@v1
|
||||||
|
id: create_release
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
with:
|
||||||
|
tag_name: ${{ github.ref }}
|
||||||
|
release_name: Ver.${{ github.ref }}
|
||||||
|
|
||||||
|
# Download(All) Artifacts to current directory
|
||||||
|
- uses: actions/download-artifact@v2-preview
|
||||||
|
|
||||||
|
# Upload to NuGet
|
||||||
|
- run: dotnet nuget push "./nuget/*.nupkg" -s https://www.nuget.org/api/v2/package -k ${{ secrets.NUGET_KEY }}
|
||||||
|
|
||||||
|
# Upload to Releases(unitypackage)
|
||||||
|
- uses: actions/upload-release-asset@v1
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
with:
|
||||||
|
upload_url: ${{ steps.create_release.outputs.upload_url }}
|
||||||
|
asset_path: ./UniTask.${{ env.GIT_TAG }}.unitypackage/UniTask.${{ env.GIT_TAG }}.unitypackage
|
||||||
|
asset_name: UniTask.${{ env.GIT_TAG }}.unitypackage
|
||||||
|
asset_content_type: application/octet-stream
|
||||||
15
.github/workflows/toc.yml
vendored
Normal file
15
.github/workflows/toc.yml
vendored
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
name: TOC Generator
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
paths:
|
||||||
|
- 'README.md'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
generateTOC:
|
||||||
|
name: TOC Generator
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: technote-space/toc-generator@v2.4.0
|
||||||
|
with:
|
||||||
|
TOC_TITLE: "## Table of Contents"
|
||||||
48
.gitignore
vendored
48
.gitignore
vendored
@@ -157,3 +157,51 @@ src/UniTask/UniTask.Tests.csproj
|
|||||||
src/UniTask/UniTask.Tests.Editor.csproj
|
src/UniTask/UniTask.Tests.Editor.csproj
|
||||||
|
|
||||||
src/UniTask/UniTask.*.unitypackage
|
src/UniTask/UniTask.*.unitypackage
|
||||||
|
|
||||||
|
src/UniTask/UniTask.Linq.csproj
|
||||||
|
|
||||||
|
src/UniTask/DOTween.Modules.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.Addressables.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.Addressables.Editor.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.Analytics.DataPrivacy.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.Recorder.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.Recorder.Editor.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.ResourceManager.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.Rider.Editor.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.ScriptableBuildPipeline.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.ScriptableBuildPipeline.Editor.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.TextMeshPro.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.TextMeshPro.Editor.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.Timeline.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.Timeline.Editor.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.VisualStudio.Editor.csproj
|
||||||
|
|
||||||
|
src/UniTask/Unity.VSCode.Editor.csproj
|
||||||
|
|
||||||
|
src/UniTask/UnityEditor.CacheServer.csproj
|
||||||
|
|
||||||
|
src/UniTask/UnityEditor.TestRunner.csproj
|
||||||
|
|
||||||
|
src/UniTask/UnityEditor.UI.csproj
|
||||||
|
|
||||||
|
src/UniTask/UnityEngine.Advertisements.csproj
|
||||||
|
|
||||||
|
src/UniTask/UnityEngine.Monetization.csproj
|
||||||
|
|
||||||
|
src/UniTask/UnityEngine.TestRunner.csproj
|
||||||
|
|
||||||
|
src/UniTask/UnityEngine.UI.csproj
|
||||||
|
|||||||
585
README.md
585
README.md
@@ -1,50 +1,106 @@
|
|||||||
# UniTask
|
UniTask
|
||||||
|
===
|
||||||
[](https://circleci.com/gh/Cysharp/UniTask)
|
[](https://github.com/Cysharp/UniTask/actions) [](https://github.com/Cysharp/UniTask/releases)
|
||||||
|
|
||||||
Provides an efficient async/await integration to Unity.
|
Provides an efficient async/await integration to Unity.
|
||||||
|
|
||||||
> UniTask was included in UniRx before v7 but now completely separated, it no dependent each other.
|
* Struct based `UniTask<T>` and custom AsyncMethodBuilder to achive zero allocation
|
||||||
|
* All Unity AsyncOperations and Coroutine to awaitable
|
||||||
|
* PlayerLoop based task(`UniTask.Yield`, `UniTask.Delay`, `UniTask.DelayFrame`, etc..) that enable to replace all coroutine operation
|
||||||
|
* MonoBehaviour Message Events and uGUI Events as awaitable/async-enumerable
|
||||||
|
* Completely run on Unity's PlayerLoop so don't use thread and run on WebGL, wasm, etc.
|
||||||
|
* Asynchronous LINQ, with Channel and AsyncReactiveProperty
|
||||||
|
* TaskTracker window to prevent memory leak
|
||||||
|
* Highly compatible behaviour with Task/ValueTask/IValueTaskSource
|
||||||
|
|
||||||
|
<!-- START doctoc generated TOC please keep comment here to allow auto update -->
|
||||||
|
<!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
|
||||||
|
## Table of Contents
|
||||||
|
|
||||||
|
- [Getting started](#getting-started)
|
||||||
|
- [Basics of UniTask and AsyncOperation](#basics-of-unitask-and-asyncoperation)
|
||||||
|
- [Cancellation and Exception handling](#cancellation-and-exception-handling)
|
||||||
|
- [Progress](#progress)
|
||||||
|
- [PlayerLoop](#playerloop)
|
||||||
|
- [async void vs async UniTaskVoid](#async-void-vs-async-unitaskvoid)
|
||||||
|
- [UniTaskTracker](#unitasktracker)
|
||||||
|
- [External Assets](#external-assets)
|
||||||
|
- [AsyncEnumerable and Async LINQ](#asyncenumerable-and-async-linq)
|
||||||
|
- [Awaitable Events](#awaitable-events)
|
||||||
|
- [Channel](#channel)
|
||||||
|
- [For Unit Testing](#for-unit-testing)
|
||||||
|
- [Pooling Configuration](#pooling-configuration)
|
||||||
|
- [API References](#api-references)
|
||||||
|
- [UPM Package](#upm-package)
|
||||||
|
- [Install via git URL](#install-via-git-url)
|
||||||
|
- [Install via OpenUPM](#install-via-openupm)
|
||||||
|
- [.NET Core](#net-core)
|
||||||
|
- [License](#license)
|
||||||
|
|
||||||
|
<!-- END doctoc generated TOC please keep comment here to allow auto update -->
|
||||||
|
|
||||||
Getting started
|
Getting started
|
||||||
---
|
---
|
||||||
Install package(`UniRx.Async.unitypackage`) is available in [UniTask/releases](https://github.com/Cysharp/UniTask/releases) page.
|
Install via [UPM package](#upm-package) or asset package(`UniTask.*.*.*.unitypackage`) available in [UniTask/releases](https://github.com/Cysharp/UniTask/releases) page.
|
||||||
|
|
||||||
```csharp
|
```csharp
|
||||||
// extension awaiter/methods can be used by this namespace
|
// extension awaiter/methods can be used by this namespace
|
||||||
using UniRx.Async;
|
using Cysharp.Threading.Tasks;
|
||||||
|
|
||||||
// You can return type as struct UniTask<T>(or UniTask), it is unity specialized lightweight alternative of Task<T>
|
// You can return type as struct UniTask<T>(or UniTask), it is unity specialized lightweight alternative of Task<T>
|
||||||
// no(or less) allocation and fast excution for zero overhead async/await integrate with Unity
|
// zero allocation and fast excution for zero overhead async/await integrate with Unity
|
||||||
async UniTask<string> DemoAsync()
|
async UniTask<string> DemoAsync()
|
||||||
{
|
{
|
||||||
// You can await Unity's AsyncObject
|
// You can await Unity's AsyncObject
|
||||||
var asset = await Resources.LoadAsync<TextAsset>("foo");
|
var asset = await Resources.LoadAsync<TextAsset>("foo");
|
||||||
|
var txt = (await UnityWebRequest.Get("https://...").SendWebRequest()).downloadHandler.text;
|
||||||
|
await SceneManager.LoadSceneAsync("scene2");
|
||||||
|
|
||||||
// .ConfigureAwait accepts progress callback
|
// .WithCancellation enables Cancel, GetCancellationTokenOnDestroy synchornizes with lifetime of GameObject
|
||||||
await SceneManager.LoadSceneAsync("scene2").ConfigureAwait(Progress.Create<float>(x => Debug.Log(x)));
|
var asset2 = await Resources.LoadAsync<TextAsset>("bar").WithCancellation(this.GetCancellationTokenOnDestroy());
|
||||||
|
|
||||||
|
// .ToUniTask accepts progress callback(and all options), Progress.Create is a lightweight alternative of IProgress<T>
|
||||||
|
var asset3 = await Resources.LoadAsync<TextAsset>("baz").ToUniTask(Progress.Create<float>(x => Debug.Log(x)));
|
||||||
|
|
||||||
// await frame-based operation like coroutine
|
// await frame-based operation like coroutine
|
||||||
await UniTask.DelayFrame(100);
|
await UniTask.DelayFrame(100);
|
||||||
|
|
||||||
// replacement of WaitForSeconds/WaitForSecondsRealtime
|
// replacement of yield return new WaitForSeconds/WaitForSecondsRealtime
|
||||||
await UniTask.Delay(TimeSpan.FromSeconds(10), ignoreTimeScale: false);
|
await UniTask.Delay(TimeSpan.FromSeconds(10), ignoreTimeScale: false);
|
||||||
|
|
||||||
// replacement of WaitForEndOfFrame(or other timing like yield return null, yield return WaitForFixedUpdate)
|
// yield any playerloop timing(PreUpdate, Update, LateUpdate, etc...)
|
||||||
await UniTask.Yield(PlayerLoopTiming.PostLateUpdate);
|
await UniTask.Yield(PlayerLoopTiming.PreLateUpdate);
|
||||||
|
|
||||||
|
// replacement of yield return null
|
||||||
|
await UniTask.Yield();
|
||||||
|
await UniTask.NextFrame();
|
||||||
|
|
||||||
|
// replacement of WaitForEndOfFrame(same as UniTask.Yield(PlayerLoopTiming.LastPostLateUpdate))
|
||||||
|
await UniTask.WaitForEndOfFrame();
|
||||||
|
|
||||||
|
// replacement of yield return new WaitForFixedUpdate(same as UniTask.Yield(PlayerLoopTiming.FixedUpdate))
|
||||||
|
await UniTask.WaitForFixedUpdate();
|
||||||
|
|
||||||
// replacement of yield return WaitUntil
|
// replacement of yield return WaitUntil
|
||||||
await UniTask.WaitUntil(() => isActive == false);
|
await UniTask.WaitUntil(() => isActive == false);
|
||||||
|
|
||||||
|
// special helper of WaitUntil
|
||||||
|
await UniTask.WaitUntilValueChanged(this, x => x.isActive);
|
||||||
|
|
||||||
// You can await IEnumerator coroutine
|
// You can await IEnumerator coroutine
|
||||||
await FooCoroutineEnumerator();
|
await FooCoroutineEnumerator();
|
||||||
|
|
||||||
// You can await standard task
|
// You can await standard task
|
||||||
await Task.Run(() => 100);
|
await Task.Run(() => 100);
|
||||||
|
|
||||||
// Multithreading, run on ThreadPool under this code(use SwitchToMainThread, same as `ObserveOnMainThreadDispatcher`)
|
// Multithreading, run on ThreadPool under this code
|
||||||
await UniTask.SwitchToThreadPool();
|
await UniTask.SwitchToThreadPool();
|
||||||
|
|
||||||
|
/* work on ThreadPool */
|
||||||
|
|
||||||
|
// return to MainThread(same as `ObserveOnMainThread` in UniRx)
|
||||||
|
await UniTask.SwitchToMainThread();
|
||||||
|
|
||||||
// get async webrequest
|
// get async webrequest
|
||||||
async UniTask<string> GetTextAsync(UnityWebRequest req)
|
async UniTask<string> GetTextAsync(UnityWebRequest req)
|
||||||
{
|
{
|
||||||
@@ -59,6 +115,9 @@ async UniTask<string> DemoAsync()
|
|||||||
// concurrent async-wait and get result easily by tuple syntax
|
// concurrent async-wait and get result easily by tuple syntax
|
||||||
var (google, bing, yahoo) = await UniTask.WhenAll(task1, task2, task3);
|
var (google, bing, yahoo) = await UniTask.WhenAll(task1, task2, task3);
|
||||||
|
|
||||||
|
// shorthand of WhenAll, tuple can await directly
|
||||||
|
var (google2, bing2, yahoo2) = await (task1, task2, task3);
|
||||||
|
|
||||||
// You can handle timeout easily
|
// You can handle timeout easily
|
||||||
await GetTextAsync(UnityWebRequest.Get("http://unity.com")).Timeout(TimeSpan.FromMilliseconds(300));
|
await GetTextAsync(UnityWebRequest.Get("http://unity.com")).Timeout(TimeSpan.FromMilliseconds(300));
|
||||||
|
|
||||||
@@ -67,47 +126,38 @@ async UniTask<string> DemoAsync()
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
`UniTask<T>`
|
Basics of UniTask and AsyncOperation
|
||||||
---
|
---
|
||||||
UniTask feature rely on C# 7.0([task-like custom async method builder feature](https://github.com/dotnet/roslyn/blob/master/docs/features/task-types.md)) so required Unity version is after `Unity 2018.3`.
|
UniTask feature rely on C# 7.0([task-like custom async method builder feature](https://github.com/dotnet/roslyn/blob/master/docs/features/task-types.md)) so required Unity version is after `Unity 2018.3`, officialy lower support version is `Unity 2018.4.13f1`.
|
||||||
|
|
||||||
Why UniTask(custom task-like object) is required? Because Task is too heavy, not matched to Unity threading(single-thread). UniTask does not use thread and SynchronizationContext because almost Unity's asynchronous object is automaticaly dispatched by Unity's engine layer. It acquires more fast and more less allocation, completely integrated with Unity.
|
Why UniTask(custom task-like object) is required? Because Task is too heavy, not matched to Unity threading(single-thread). UniTask does not use thread and SynchronizationContext/ExecutionContext because almost Unity's asynchronous object is automaticaly dispatched by Unity's engine layer. It acquires more fast and more less allocation, completely integrated with Unity.
|
||||||
|
|
||||||
> More details, please see this slide: [Deep Dive async/await in Unity with UniTask(EN)
|
You can await `AsyncOperation`, `ResourceRequest`, `AssetBundleRequest`, `AssetBundleCreateRequest`, `UnityWebRequestAsyncOperation`, `IEnumerator` and others when `using Cysharp.Threading.Tasks;`.
|
||||||
](https://www.slideshare.net/neuecc/deep-dive-asyncawait-in-unity-with-unitasken)
|
|
||||||
|
|
||||||
You can await `AsyncOperation`, `ResourceRequest`, `UnityWebRequestAsyncOperation`, `IEnumerator` and others when using `UniRx.Async`.
|
UniTask provides three pattern of extension methods.
|
||||||
|
|
||||||
`UniTask.Delay`, `UniTask.Yield`, `UniTask.Timeout` that is frame-based timer operators(no uses thread so works on WebGL publish) driven by custom PlayerLoop(Unity 2018 experimental feature). In default, UniTask initialize automatically when application begin, but it is override all. If you want to append PlayerLoop, please call `PlayerLoopHelper.Initialize(ref yourCustomizedPlayerLoop)` manually.
|
|
||||||
|
|
||||||
> Before Unity 2019.3, Unity does not have `PlayerLooop.GetCurrentPlayerLoop` so you can't use with Unity ECS package in default. If you want to use with ECS and before Unity 2019.3, you can use this hack below.
|
|
||||||
|
|
||||||
```csharp
|
```csharp
|
||||||
// Get ECS Loop.
|
* await asyncOperation;
|
||||||
var playerLoop = ScriptBehaviourUpdateOrder.CurrentPlayerLoop;
|
* .WithCancellation(CancellationToken);
|
||||||
|
* .ToUniTask(IProgress, PlayerLoopTiming, CancellationToken);
|
||||||
// Setup UniTask's PlayerLoop.
|
|
||||||
PlayerLoopHelper.Initialize(ref playerLoop);
|
|
||||||
```
|
```
|
||||||
|
|
||||||
`UniTask.WhenAll`, `UniTask.WhenAny` is like Task.WhenAll/WhenAny but return type is more useful.
|
`WithCancellation` is a simple version of `ToUniTask`, both returns `UniTask`. Details of cancellation, see: [Cancellation and Exception handling](#cancellation-and-exception-handling) section.
|
||||||
|
|
||||||
`UniTask.ctor(Func<UniTask>)` is like the embeded [`AsyncLazy<T>`](https://blogs.msdn.microsoft.com/pfxteam/2011/01/15/asynclazyt/)
|
> Note: WithCancellation is returned from native timing of PlayerLoop but ToUniTask is returned from specified PlayerLoopTiming. Details of timing, see: [PlayerLoop](#playerloop) section.
|
||||||
|
|
||||||
|
The type of `UniTask` can use utility like `UniTask.WhenAll`, `UniTask.WhenAny`. It is like Task.WhenAll/WhenAny but return type is more useful, returns value tuple so can deconsrtuct each result and pass multiple type.
|
||||||
|
|
||||||
```csharp
|
```csharp
|
||||||
public class SceneAssets
|
public class SceneAssets
|
||||||
{
|
{
|
||||||
public readonly UniTask<Sprite> Front;
|
|
||||||
public readonly UniTask<Sprite> Background;
|
|
||||||
public readonly UniTask<Sprite> Effect;
|
|
||||||
|
|
||||||
public SceneAssets()
|
public SceneAssets()
|
||||||
{
|
{
|
||||||
// ctor(Func) overload is AsyncLazy, initialized once when await.
|
// parallel load.
|
||||||
// and after it, await returns zero-allocation value immediately.
|
var (a, b, c) = await UniTask.WhenAll(
|
||||||
Front = new UniTask<Sprite>(() => LoadAsSprite("foo"));
|
LoadAsSprite("foo"),
|
||||||
Background = new UniTask<Sprite>(() => LoadAsSprite("bar"));
|
LoadAsSprite("bar"),
|
||||||
Effect = new UniTask<Sprite>(() => LoadAsSprite("baz"));
|
LoadAsSprite("baz"));
|
||||||
}
|
}
|
||||||
|
|
||||||
async UniTask<Sprite> LoadAsSprite(string path)
|
async UniTask<Sprite> LoadAsSprite(string path)
|
||||||
@@ -133,13 +183,32 @@ public UniTask<int> WrapByUniTaskCompletionSource()
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
You can convert Task -> UniTask: `AsUniTask`, `UniTask` -> `UniTask<AsyncUnit>`: `AsAsyncUnitUniTask`(this is useful to use WhenAll/WhenAny), `UniTask<T>` -> `UniTask`: `AsUniTask`.
|
You can convert Task -> UniTask: `AsUniTask`, `UniTask` -> `UniTask<AsyncUnit>`: `AsAsyncUnitUniTask`, `UniTask<T>` -> `UniTask`: `AsUniTask`. `UniTask<T>` -> `UniTask`'s conversion cost is free.
|
||||||
|
|
||||||
If you want to convert async to coroutine, you can use `UniTask.ToCoroutine`, this is useful to use only allow coroutine system.
|
If you want to convert async to coroutine, you can use `.ToCoroutine()`, this is useful to use only allow coroutine system.
|
||||||
|
|
||||||
|
UniTask can not await twice. This is a similar constraint to the [ValueTask/IValueTaskSource](https://docs.microsoft.com/en-us/dotnet/api/system.threading.tasks.valuetask-1?view=netcore-3.1) introduced in .NET Standard 2.1.
|
||||||
|
|
||||||
|
> The following operations should never be performed on a ValueTask<TResult> instance:
|
||||||
|
>
|
||||||
|
> * Awaiting the instance multiple times.
|
||||||
|
> * Calling AsTask multiple times.
|
||||||
|
> * Using .Result or .GetAwaiter().GetResult() when the operation hasn't yet completed, or using them multiple times.
|
||||||
|
> * Using more than one of these techniques to consume the instance.
|
||||||
|
>
|
||||||
|
> If you do any of the above, the results are undefined.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
var task = UniTask.DelayFrame(10);
|
||||||
|
await task;
|
||||||
|
await task; // NG, throws Exception
|
||||||
|
```
|
||||||
|
|
||||||
|
Store to the class field, you can use `UniTask.Lazy` that gurantee call multipletimes. `.Prevent()` allows for multiple calls (internally cached results). This is useful when multiple calls in a function scope.
|
||||||
|
|
||||||
Cancellation and Exception handling
|
Cancellation and Exception handling
|
||||||
---
|
---
|
||||||
Some UniTask factory methods have `CancellationToken cancellation = default(CancellationToken)` parameter. Andalso some async operation for unity have `ConfigureAwait(..., CancellationToken cancellation = default(CancellationToken))` extension methods.
|
Some UniTask factory methods have `CancellationToken cancellationToken = default` parameter. Andalso some async operation for unity have `WithCancellation(CancellationToken)` and `ToUniTask(..., CancellationToken cancellation = default)` extension methods.
|
||||||
|
|
||||||
You can pass `CancellationToken` to parameter by standard [`CancellationTokenSource`](https://docs.microsoft.com/en-us/dotnet/api/system.threading.cancellationtokensource).
|
You can pass `CancellationToken` to parameter by standard [`CancellationTokenSource`](https://docs.microsoft.com/en-us/dotnet/api/system.threading.cancellationtokensource).
|
||||||
|
|
||||||
@@ -151,7 +220,7 @@ cancelButton.onClick.AddListener(() =>
|
|||||||
cts.Cancel();
|
cts.Cancel();
|
||||||
});
|
});
|
||||||
|
|
||||||
await UnityWebRequest.Get("http://google.co.jp").SendWebRequest().ConfigureAwait(cancellation: cts.Token);
|
await UnityWebRequest.Get("http://google.co.jp").SendWebRequest().WithCancellation(cts.Token);
|
||||||
|
|
||||||
await UniTask.DelayFrame(1000, cancellationToken: cts.Token);
|
await UniTask.DelayFrame(1000, cancellationToken: cts.Token);
|
||||||
```
|
```
|
||||||
@@ -165,7 +234,7 @@ await UniTask.DelayFrame(1000, cancellationToken: this.GetCancellationTokenOnDes
|
|||||||
|
|
||||||
When detect cancellation, all methods throws `OperationCanceledException` and propagate to upstream. `OperationCanceledException` is special exception, if not handled this exception, finally it is propagated to `UniTaskScheduler.UnobservedTaskException`.
|
When detect cancellation, all methods throws `OperationCanceledException` and propagate to upstream. `OperationCanceledException` is special exception, if not handled this exception, finally it is propagated to `UniTaskScheduler.UnobservedTaskException`.
|
||||||
|
|
||||||
Default behaviour of received unhandled exception is write log as warning. Log level can change by `UniTaskScheduler.UnobservedExceptionWriteLogType`. If you want to change custom beavhiour, set action to `UniTaskScheduler.UnobservedTaskException.`
|
Default behaviour of received unhandled exception is write log as exception. Log level can change by `UniTaskScheduler.UnobservedExceptionWriteLogType`. If you want to change custom beavhiour, set action to `UniTaskScheduler.UnobservedTaskException.`
|
||||||
|
|
||||||
If you want to cancel behaviour in async UniTask method, throws `OperationCanceledException` manually.
|
If you want to cancel behaviour in async UniTask method, throws `OperationCanceledException` manually.
|
||||||
|
|
||||||
@@ -204,23 +273,23 @@ if (isCanceled)
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
Note: Only suppress throws if you call it directly into the most source method.
|
Note: Only suppress throws if you call it directly into the most source method. Otherwise, the return value will be converted, but the entire pipeline will not be suppressed throws.
|
||||||
|
|
||||||
Progress
|
Progress
|
||||||
---
|
---
|
||||||
Some async operation for unity have `ConfigureAwait(IProgress<float> progress = null, ...)` extension methods.
|
Some async operation for unity have `ToUniTask(IProgress<float> progress = null, ...)` extension methods.
|
||||||
|
|
||||||
```csharp
|
```csharp
|
||||||
var progress = Progress.Create<float>(x => Debug.Log(x));
|
var progress = Progress.Create<float>(x => Debug.Log(x));
|
||||||
|
|
||||||
var request = await UnityWebRequest.Get("http://google.co.jp")
|
var request = await UnityWebRequest.Get("http://google.co.jp")
|
||||||
.SendWebRequest()
|
.SendWebRequest()
|
||||||
.ConfigureAwait(progress: progress);
|
.ToUniTask(progress: progress);
|
||||||
```
|
```
|
||||||
|
|
||||||
Should not use `new System.Progress<T>`, because it allocate every times. Use `UniRx.Async.Progress` instead. Progress factory has two methods, `Create` and `CreateOnlyValueChanged`. `CreateOnlyValueChanged` calls only when progress value changed. Should not use `new System.Progress<T>`, it allocate every times.
|
You should not use standard `new System.Progress<T>`, because it causes allocation every times. Use `Cysharp.Threading.Tasks.Progress` instead. This progress factory has two methods, `Create` and `CreateOnlyValueChanged`. `CreateOnlyValueChanged` calls only when progress value changed.
|
||||||
|
|
||||||
Implements interface is more better.
|
Implements IProgress interface to caller is more better, there is no allocation of lambda.
|
||||||
|
|
||||||
```csharp
|
```csharp
|
||||||
public class Foo : MonoBehaviour, IProgress<float>
|
public class Foo : MonoBehaviour, IProgress<float>
|
||||||
@@ -234,16 +303,135 @@ public class Foo : MonoBehaviour, IProgress<float>
|
|||||||
{
|
{
|
||||||
var request = await UnityWebRequest.Get("http://google.co.jp")
|
var request = await UnityWebRequest.Get("http://google.co.jp")
|
||||||
.SendWebRequest()
|
.SendWebRequest()
|
||||||
.ConfigureAwait(progress: this); // pass this
|
.ToUniTask(progress: this); // pass this
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
PlayerLoop
|
||||||
|
---
|
||||||
|
UniTask is run on custom [PlayerLoop](https://docs.unity3d.com/ScriptReference/LowLevel.PlayerLoop.html). UniTask's playerloop based method(such as `Delay`, `DelayFrame`, `asyncOperation.ToUniTask`, etc...) accepts this `PlayerLoopTiming`.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
public enum PlayerLoopTiming
|
||||||
|
{
|
||||||
|
Initialization = 0,
|
||||||
|
LastInitialization = 1,
|
||||||
|
|
||||||
|
EarlyUpdate = 2,
|
||||||
|
LastEarlyUpdate = 3,
|
||||||
|
|
||||||
|
FixedUpdate = 4,
|
||||||
|
LastFixedUpdate = 5,
|
||||||
|
|
||||||
|
PreUpdate = 6,
|
||||||
|
LastPreUpdate = 7,
|
||||||
|
|
||||||
|
Update = 8,
|
||||||
|
LastUpdate = 9,
|
||||||
|
|
||||||
|
PreLateUpdate = 10,
|
||||||
|
LastPreLateUpdate = 11,
|
||||||
|
|
||||||
|
PostLateUpdate = 12,
|
||||||
|
LastPostLateUpdate = 13
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
It indicates when to run, you can check [PlayerLoopList.md](https://gist.github.com/neuecc/bc3a1cfd4d74501ad057e49efcd7bdae) to Unity's default playerloop and injected UniTask's custom loop.
|
||||||
|
|
||||||
|
`PlayerLoopTiming.Update` is similar as `yield return null` in coroutine, but it is called before Update(Update and uGUI events(button.onClick, etc...) are called on `ScriptRunBehaviourUpdate`, yield return null is called on `ScriptRunDelayedDynamicFrameRate`). `PlayerLoopTiming.FixedUpdate` is similar as `WaitForFixedUpdate`, `PlayerLoopTiming.LastPostLateUpdate` is similar as `WaitForEndOfFrame` in coroutine.
|
||||||
|
|
||||||
|
`yield return null` and `UniTask.Yield` is similar but different. `yield return null` always return next frame but `UniTask.Yield` return next called, that is, call `UniTask.Yield(PlayerLoopTiming.Update)` on `PreUpdate`, it returns same frame. `UniTask.NextFrame()` gurantees return next frame, this would be expected to behave exactly the same as `yield return null`.
|
||||||
|
|
||||||
|
> UniTask.Yield(without CancellationToken) is a special type, returns `YieldAwaitable` and run on YieldRunner. It is most lightweight and faster.
|
||||||
|
|
||||||
|
AsyncOperation is returned from native timing. For example, await `SceneManager.LoadSceneAsync` is returned from `EarlyUpdate.UpdatePreloading` and after called, loaded scene called from `EarlyUpdate.ScriptRunDelayedStartupFrame`. Also `await UnityWebRequest` is returned from `EarlyUpdate.ExecuteMainThreadJobs`.
|
||||||
|
|
||||||
|
In UniTask, await directly and `WithCancellation` use native timing, `ToUniTask` use specified timing. This is usually not a particular problem, but with `LoadSceneAsync`, causes different order of Start and continuation after await. so recommend not to use `LoadSceneAsync.ToUniTask`.
|
||||||
|
|
||||||
|
In stacktrace, you can check where is running in playerloop.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
In default, UniTask's PlayerLoop is initialized at `[RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.BeforeSceneLoad)]`.
|
||||||
|
|
||||||
|
The order in which methods are called in BeforeSceneLoad is indeterminate, so if you want to use UniTask in other BeforeSceneLoad methods, you should try to initialize it before this.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// AfterAssembliesLoaded is called before BeforeSceneLoad
|
||||||
|
[RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.AfterAssembliesLoaded)]
|
||||||
|
public static void InitUniTaskLoop()
|
||||||
|
{
|
||||||
|
var loop = PlayerLoop.GetCurrentPlayerLoop();
|
||||||
|
Cysharp.Threading.Tasks.PlayerLoopHelper.Initialize(ref loop);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
If you import Unity's `Entities` package, that reset custom player loop to default at `BeforeSceneLoad` and inject ECS's loop. When Unity call ECS's inject method after UniTask's initialize method, UniTask will no longer work.
|
||||||
|
|
||||||
|
To solve this issue, you can re-initialize UniTask PlayerLoop after ECS initialized.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// Get ECS Loop.
|
||||||
|
var playerLoop = ScriptBehaviourUpdateOrder.CurrentPlayerLoop;
|
||||||
|
|
||||||
|
// Setup UniTask's PlayerLoop.
|
||||||
|
PlayerLoopHelper.Initialize(ref playerLoop);
|
||||||
|
```
|
||||||
|
|
||||||
|
async void vs async UniTaskVoid
|
||||||
|
---
|
||||||
|
`async void` is a standard C# taks system so does not run on UniTask systems. It is better not to use. `async UniTaskVoid` is a lightweight version of `async UniTask` because it does not have awaitable completion and report error immediately to `UniTaskScheduler.UnobservedTaskException`. If you don't require to await it(fire and forget), use `UniTaskVoid` is better. Unfortunately to dismiss warning, require to using with `Forget()`.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
public async UniTaskVoid FireAndForgetMethod()
|
||||||
|
{
|
||||||
|
// do anything...
|
||||||
|
await UniTask.Yield();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Caller()
|
||||||
|
{
|
||||||
|
FireAndForgetMethod().Forget();
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Also UniTask have `Forget` method, it is similar with UniTaskVoid and same effects with it. However still UniTaskVoid is more efficient if completely do not use await。
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
public async UniTask DoAsync()
|
||||||
|
{
|
||||||
|
// do anything...
|
||||||
|
await UniTask.Yield();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Caller()
|
||||||
|
{
|
||||||
|
DoAsync().Forget();
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Using async lambda in register event, it is used `async void`. To avoid it, you can use `UniTask.Action` or `UniTask.UnityAction` that creates delegate via `async UniTaskVoid` lambda.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
Action actEvent;
|
||||||
|
UnityAction unityEvent; // especially used in uGUI
|
||||||
|
|
||||||
|
// Bad: async void
|
||||||
|
actEvent += async () => { };
|
||||||
|
unityEvent += async () => { };
|
||||||
|
|
||||||
|
// Ok: create Action delegate by lambda
|
||||||
|
actEvent += UniTask.Action(async () => { await UniTask.Yield(); });
|
||||||
|
unityEvent += UniTask.UnityAction(async () => { await UniTask.Yield(); });
|
||||||
|
```
|
||||||
|
|
||||||
UniTaskTracker
|
UniTaskTracker
|
||||||
---
|
---
|
||||||
useful for check(leak) UniTasks. You can open tracker window in `Window -> UniTask Tracker`.
|
useful for check(leak) UniTasks. You can open tracker window in `Window -> UniTask Tracker`.
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
* Enable AutoReload(Toggle) - Reload automatically.
|
* Enable AutoReload(Toggle) - Reload automatically.
|
||||||
* Reload - Reload view.
|
* Reload - Reload view.
|
||||||
@@ -253,38 +441,89 @@ useful for check(leak) UniTasks. You can open tracker window in `Window -> UniTa
|
|||||||
|
|
||||||
For debug use, enable tracking and capture stacktrace is useful but it it decline performance. Recommended usage is enable both to find task leak, and when done, finally disable both.
|
For debug use, enable tracking and capture stacktrace is useful but it it decline performance. Recommended usage is enable both to find task leak, and when done, finally disable both.
|
||||||
|
|
||||||
Reusable Promises
|
External Assets
|
||||||
---
|
---
|
||||||
Some UniTask factory can reuse to reduce allocation. The list is `Yield`, `Delay`, `DelayFrame`, `WaitUntil`, `WaitWhile`, `WaitUntilValueChanged`.
|
In default, UniTask supports DOTween and Addressables(`AsyncOperationHandle` and `AsyncOpereationHandle<T>` as awaitable).
|
||||||
|
|
||||||
|
For DOTween support, require to `com.demigiant.dotween` import from [OpenUPM](https://openupm.com/packages/com.demigiant.dotween/) or define `UNITASK_DOTWEEN_SUPPORT` to enable it.
|
||||||
|
|
||||||
```csharp
|
```csharp
|
||||||
var reusePromise = UniTask.DelayFrame(10);
|
// sequential
|
||||||
|
await transform.DOMoveX(2, 10);
|
||||||
|
await transform.DOMoveZ(5, 20);
|
||||||
|
|
||||||
for (int i = 0; i < 10; i++)
|
// parallel with cancellation
|
||||||
|
var ct = this.GetCancellationTokenOnDestroy();
|
||||||
|
|
||||||
|
await UniTask.WhenAll(
|
||||||
|
transform.DOMoveX(10, 3).WithCancellation(ct),
|
||||||
|
transform.DOScale(10, 3).WithCancellation(ct));
|
||||||
|
```
|
||||||
|
|
||||||
|
AsyncEnumerable and Async LINQ
|
||||||
|
---
|
||||||
|
Unity 2020.2.0a12 supports C# 8.0 so you can use `await foreach`. This is the new Update notation in async era.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// Unity 2020.2.0a12, C# 8.0
|
||||||
|
await foreach (var _ in UniTaskAsyncEnumerable.EveryUpdate(token))
|
||||||
{
|
{
|
||||||
await reusePromise;
|
Debug.Log("Update() " + Time.frameCount);
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
awaitable Events
|
In a C# 7.3 environment, you can use the `ForEachAsync` method to work in almost the same way.
|
||||||
---
|
|
||||||
Unity events can await like `OnClickAsync`, `OnCollisionEnterAsync`. It can use by `UniRx.Async.Triggers`.
|
|
||||||
|
|
||||||
```csharp
|
```csharp
|
||||||
using UniRx.Async.Triggers;
|
// C# 7.3(Unity 2018.3~)
|
||||||
|
await UniTaskAsyncEnumerable.EveryUpdate(token).ForEachAsync(_ =>
|
||||||
async UniTask TripleClick(CancellationToken token)
|
|
||||||
{
|
{
|
||||||
await button.OnClickAsync(token);
|
Debug.Log("Update() " + Time.frameCount);
|
||||||
await button.OnClickAsync(token);
|
});
|
||||||
await button.OnClickAsync(token);
|
```
|
||||||
|
|
||||||
|
UniTaskAsyncEnumerable implements asynchronous LINQ, similar to LINQ in `IEnumerable<T>` or Rx in `IObservable<T>`. All standard LINQ query operators can be applied to asynchronous streams. For example, the following code shows how to apply a Where filter to a button-click asynchronous stream that runs once every two clicks.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
await okButton.OnClickAsAsyncEnumerable().Where((x, i) => i % 2 == 0).ForEachAsync(_ =>
|
||||||
|
{
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
Async LINQ is enabled when `using Cysharp.Threading.Tasks.Linq;`, and `UniTaskAsyncEnumerable` is defined in `UniTask.Linq` asmdef.
|
||||||
|
|
||||||
|
It's closer to UniRx (Reactive Extensions), but UniTaskAsyncEnumerable is a pull-based asynchronous stream, whereas Rx was a push-based asynchronous stream. Note that although similar, the characteristics are different and the details behave differently along with them.
|
||||||
|
|
||||||
|
`UniTaskAsyncEnumerable` is the entry point like `Enumerbale`. In addition to the standard query operators, there are other generators for Unity such as `EveryUpdate`, `Timer`, `TimerFrame`, `Interval`, `IntervalFrame`, and `EveryValueChanged`. And also added additional UniTask original query operators like `Append`, `Prepend`, `DistinctUntilChanged`, `ToHashSet`, `Buffer`, `CombineLatest`, `Do`, `Never`, `ForEachAsync`, `Pairwise`, `Publish`, `Queue`, `Return`, `SkipUntilCanceled`, `TakeUntilCanceled`, `TakeLast`.
|
||||||
|
|
||||||
|
The method with Func as an argument has three additional overloads, `***Await`, `***AwaitWithCancellation`.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
Select(Func<T, TR> selector)
|
||||||
|
SelectAwait(Func<T, UniTask<TR>> selector)
|
||||||
|
SelectAwaitWithCancellation(Func<T, CancellationToken, UniTask<TR>> selector)
|
||||||
|
```
|
||||||
|
|
||||||
|
If you want to use the `async` method inside the func, use the `***Await` or `***AwaitWithCancellation`.
|
||||||
|
|
||||||
|
Awaitable Events
|
||||||
|
---
|
||||||
|
All uGUI component implements `***AsAsyncEnumerable` to convert asynchronous streams of events.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
async UniTask TripleClick()
|
||||||
|
{
|
||||||
|
// In default, used button.GetCancellationTokenOnDestroy to manage lieftime of async
|
||||||
|
await button.OnClickAsync();
|
||||||
|
await button.OnClickAsync();
|
||||||
|
await button.OnClickAsync();
|
||||||
Debug.Log("Three times clicked");
|
Debug.Log("Three times clicked");
|
||||||
}
|
}
|
||||||
|
|
||||||
// more efficient way
|
// more efficient way
|
||||||
async UniTask TripleClick(CancellationToken token)
|
async UniTask TripleClick()
|
||||||
{
|
{
|
||||||
using (var handler = button.GetAsyncClickEventHandler(token))
|
using (var handler = button.GetAsyncClickEventHandler())
|
||||||
{
|
{
|
||||||
await handler.OnClickAsync();
|
await handler.OnClickAsync();
|
||||||
await handler.OnClickAsync();
|
await handler.OnClickAsync();
|
||||||
@@ -292,11 +531,126 @@ async UniTask TripleClick(CancellationToken token)
|
|||||||
Debug.Log("Three times clicked");
|
Debug.Log("Three times clicked");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// use async LINQ
|
||||||
|
async UniTask TripleClick(CancellationToken token)
|
||||||
|
{
|
||||||
|
await button.OnClickAsAsyncEnumerable().Take(3).Last();
|
||||||
|
Debug.Log("Three times clicked");
|
||||||
|
}
|
||||||
|
|
||||||
|
// use async LINQ2
|
||||||
|
async UniTask TripleClick(CancellationToken token)
|
||||||
|
{
|
||||||
|
await button.OnClickAsAsyncEnumerable().Take(3).ForEachAsync(_ =>
|
||||||
|
{
|
||||||
|
Debug.Log("Every clicked");
|
||||||
|
});
|
||||||
|
Debug.Log("Three times clicked, complete.");
|
||||||
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
async void vs async UniTask/UniTaskVoid
|
All MonoBehaviour message events can convert async-streams by `AsyncTriggers` that can enable by `using Cysharp.Threading.Tasks.Triggers;`.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
using Cysharp.Threading.Tasks.Triggers;
|
||||||
|
|
||||||
|
async UniTaskVoid MonitorCollision()
|
||||||
|
{
|
||||||
|
await gameObject.OnCollisionEnterAsync();
|
||||||
|
Debug.Log("Collision Enter");
|
||||||
|
/* do anything */
|
||||||
|
|
||||||
|
await gameObject.OnCollisionExitAsync();
|
||||||
|
Debug.Log("Collision Exit");
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Similar as uGUI event, AsyncTrigger can get by `GetAsync***Trigger` and trigger it self is UniTaskAsyncEnumerable.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// use await multiple times, get AsyncTriggerHandler is more efficient.
|
||||||
|
using(var trigger = this.GetOnCollisionEnterAsyncHandler())
|
||||||
|
{
|
||||||
|
await OnCollisionEnterAsync();
|
||||||
|
await OnCollisionEnterAsync();
|
||||||
|
await OnCollisionEnterAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
// every moves.
|
||||||
|
await this.GetAsyncMoveTrigger().ForEachAsync(axisEventData =>
|
||||||
|
{
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
`AsyncReactiveProperty`, `AsyncReadOnlyReactiveProperty` is UniTask version of UniTask's ReactiveProperty.
|
||||||
|
|
||||||
|
`BindTo` extension method of `IUniTaskAsyncEnumerable<T>` for binding asynchronous stream values to Unity components(Text/Selectable/TMP/Text).
|
||||||
|
|
||||||
|
A pull-type asynchronous stream does not get the next values until the asynchronous processing in the sequence is complete. This could spill data from push-type events such as buttons.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// can not get click event during 3 seconds complete.
|
||||||
|
await button.OnClickAsAsyncEnumerable().ForEachAwaitAsync()
|
||||||
|
{
|
||||||
|
await UniTask.Delay(TimeSpan.FromSeconds(3));
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
It is useful(prevent double-click) but not useful in sometimes.
|
||||||
|
|
||||||
|
Using `Queue()` method, which will also queue events during asynchronous processing.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
// queued message in asynchronous processing
|
||||||
|
await button.OnClickAsAsyncEnumerable().Queue().ForEachAwaitAsync()
|
||||||
|
{
|
||||||
|
await UniTask.Delay(TimeSpan.FromSeconds(3));
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Channel
|
||||||
---
|
---
|
||||||
`async void` is standard C# system so does not run on UniTask systems. It is better not to use. `async UniTaskVoid` is lightweight version of `async UniTask` because it does not have awaitable completion. If you don't require to await it(fire and forget), use `UniTaskVoid` is better. Unfortunately to dismiss warning, require to using with `Forget()`.
|
`Channel` is same as [System.Threading.Tasks.Channels](https://docs.microsoft.com/ja-jp/dotnet/api/system.threading.channels?view=netcore-3.1) that is similar as GoLang Channel.
|
||||||
|
|
||||||
|
Currently only supports multiple-producer, single-consumer unbounded channel. It can create by `Channel.CreateSingleConsumerUnbounded<T>()`.
|
||||||
|
|
||||||
|
For producer(`.Writer`), `TryWrite` to push value and `TryComplete` to complete channel. For consumer(`.Reader`), `TryRead`, `WaitToReadAsync`, `ReadAsync`, `Completion` and `ReadAllAsync` to read queued messages.
|
||||||
|
|
||||||
|
`ReadAllAsync` returns `IUniTaskAsyncEnumerable<T>` so query LINQ operators. Reader only allows single-consumer but use `.Publish()` query operator to enable multicast message. For example, make pub/sub utility.
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
public class AsyncMessageBroker<T> : IDisposable
|
||||||
|
{
|
||||||
|
Channel<T> channel;
|
||||||
|
|
||||||
|
IConnectableUniTaskAsyncEnumerable<T> multicastSource;
|
||||||
|
IDisposable connection;
|
||||||
|
|
||||||
|
public AsyncMessageBroker()
|
||||||
|
{
|
||||||
|
channel = Channel.CreateSingleConsumerUnbounded<T>();
|
||||||
|
multicastSource = channel.Reader.ReadAllAsync().Publish();
|
||||||
|
connection = multicastSource.Connect(); // Publish returns IConnectableUniTaskAsyncEnumerable.
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Publish(T value)
|
||||||
|
{
|
||||||
|
channel.Writer.TryWrite(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
public IUniTaskAsyncEnumerable<T> Subscribe()
|
||||||
|
{
|
||||||
|
return multicastSource;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
channel.Writer.TryComplete();
|
||||||
|
connection.Dispose();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
For Unit Testing
|
For Unit Testing
|
||||||
---
|
---
|
||||||
@@ -323,36 +677,91 @@ public IEnumerator DelayIgnore() => UniTask.ToCoroutine(async () =>
|
|||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
Method List
|
UniTask itself's unit test is written by Unity Test Runner and [Cysharp/RuntimeUnitTestToolkit](https://github.com/Cysharp/RuntimeUnitTestToolkit) to check on CI and IL2CPP working.
|
||||||
|
|
||||||
|
Pooling Configuration
|
||||||
---
|
---
|
||||||
|
UniTask is aggressively caching async promise object to achive zero allocation. In default, cache all promises but you can configure `TaskPool.SetMaxPoolSize` to your value, the value indicates cache size per type. `TaskPool.GetCacheSizeInfo` returns current cached object in pool.
|
||||||
|
|
||||||
```csharp
|
```csharp
|
||||||
UniTask.WaitUntil
|
foreach (var (type, size) in TaskPool.GetCacheSizeInfo())
|
||||||
UniTask.WaitWhile
|
{
|
||||||
UniTask.WaitUntilValueChanged
|
Debug.Log(type + ":" + size);
|
||||||
UniTask.SwitchToThreadPool
|
}
|
||||||
UniTask.SwitchToTaskPool
|
|
||||||
UniTask.SwitchToMainThread
|
|
||||||
UniTask.SwitchToSynchronizationContext
|
|
||||||
UniTask.Yield
|
|
||||||
UniTask.Run
|
|
||||||
UniTask.Lazy
|
|
||||||
UniTask.Void
|
|
||||||
UniTask.ConfigureAwait
|
|
||||||
UniTask.DelayFrame
|
|
||||||
UniTask.Delay(..., bool ignoreTimeScale = false, ...) parameter
|
|
||||||
```
|
```
|
||||||
|
|
||||||
|
API References
|
||||||
|
---
|
||||||
|
UniTask's API References is hosted at [cysharp.github.io/UniTask](https://cysharp.github.io/UniTask/api/Cysharp.Threading.Tasks.html) by [DocFX](https://dotnet.github.io/docfx/) and [Cysharp/DocfXTemplate](https://github.com/Cysharp/DocfxTemplate).
|
||||||
|
|
||||||
|
For example, UniTask's factory methods can see at [UniTask#methods](https://cysharp.github.io/UniTask/api/Cysharp.Threading.Tasks.UniTask.html#methods-1). UniTaskAsyncEnumerable's factory/extension methods can see at [UniTaskAsyncEnumerable#methods](https://cysharp.github.io/UniTask/api/Cysharp.Threading.Tasks.Linq.UniTaskAsyncEnumerable.html#methods-1).
|
||||||
|
|
||||||
UPM Package
|
UPM Package
|
||||||
---
|
---
|
||||||
After Unity 2019.3.4f1, Unity 2020.1a21, that support path query parameter of git package. You can add `https://github.com/Cysharp/UniTask.git?path=Assets/UniRx.Async` to Package Manager
|
### Install via git URL
|
||||||
|
|
||||||
|
After Unity 2019.3.4f1, Unity 2020.1a21, that support path query parameter of git package. You can add `https://github.com/Cysharp/UniTask.git?path=src/UniTask/Assets/Plugins/UniTask` to Package Manager
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
or add `"com.cysharp.unitask": "https://github.com/Cysharp/UniTask.git?path=Assets/UniRx.Async"` to `Packages/manifest.json`.
|
or add `"com.cysharp.unitask": "https://github.com/Cysharp/UniTask.git?path=src/UniTask/Assets/Plugins/UniTask"` to `Packages/manifest.json`.
|
||||||
|
|
||||||
If you want to set a target version, UniTask is using `*.*.*` release tag so you can specify a version like `#1.3.0`. For example `https://github.com/Cysharp/UniTask.git?path=Assets/UniRx.Async#1.3.1`.
|
If you want to set a target version, UniTask is using `*.*.*` release tag so you can specify a version like `#2.0.13`. For example `https://github.com/Cysharp/UniTask.git?path=src/UniTask/Assets/Plugins/UniTask#2.0.13`.
|
||||||
|
|
||||||
|
### Install via OpenUPM
|
||||||
|
|
||||||
|
The package is available on the [openupm registry](https://openupm.com). It's recommended to install it via [openupm-cli](https://github.com/openupm/openupm-cli).
|
||||||
|
|
||||||
|
```
|
||||||
|
openupm add com.cysharp.unitask
|
||||||
|
```
|
||||||
|
|
||||||
|
.NET Core
|
||||||
|
---
|
||||||
|
For .NET Core, use NuGet.
|
||||||
|
|
||||||
|
> PM> Install-Package [UniTask](https://www.nuget.org/packages/UniTask)
|
||||||
|
|
||||||
|
UniTask of .NET Core version is a subset of Unity UniTask, removed PlayerLoop dependent methods.
|
||||||
|
|
||||||
|
It runs at higher performance than the standard Task/ValueTask, but you should be careful to ignore the ExecutionContext/SynchronizationContext when using it. `AysncLocal` also does not work because it ignores ExecutionContext.
|
||||||
|
|
||||||
|
If you use UniTask internally, but provide ValueTask as an external API, you can write like the following(Inspired by [PooledAwait](https://github.com/mgravell/PooledAwait)).
|
||||||
|
|
||||||
|
```csharp
|
||||||
|
public class ZeroAllocAsyncAwaitInDotNetCore
|
||||||
|
{
|
||||||
|
public ValueTask<int> DoAsync(int x, int y)
|
||||||
|
{
|
||||||
|
return Core(this, x, y);
|
||||||
|
|
||||||
|
static async UniTask<int> Core(ZeroAllocAsyncAwaitInDotNetCore self, int x, int y)
|
||||||
|
{
|
||||||
|
// do anything...
|
||||||
|
await Task.Delay(TimeSpan.FromSeconds(x + y));
|
||||||
|
await UniTask.Yield();
|
||||||
|
|
||||||
|
return 10;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// UniTask does not return to original SynchronizationContext but you can use helper `ReturnToCurrentSynchronizationContext`.
|
||||||
|
public ValueTask TestAsync()
|
||||||
|
{
|
||||||
|
await using (UniTask.ReturnToCurrentSynchronizationContext())
|
||||||
|
{
|
||||||
|
await UniTask.SwitchToThreadPool();
|
||||||
|
// do anything..
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
.NET Core version is intended to allow users to use UniTask as an interface when sharing code with Unity (such as [Cysharp/MagicOnion](https://github.com/Cysharp/MagicOnion/)). .NET Core version of UniTask enables smooth code sharing.
|
||||||
|
|
||||||
|
Utility methods such as WhenAll which is equivalent to UniTask are provided as [Cysharp/ValueTaskSupplement](https://github.com/Cysharp/ValueTaskSupplement).
|
||||||
|
|
||||||
License
|
License
|
||||||
---
|
---
|
||||||
|
|||||||
10
docs/.gitignore
vendored
Normal file
10
docs/.gitignore
vendored
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
###############
|
||||||
|
# folder #
|
||||||
|
###############
|
||||||
|
/**/DROP/
|
||||||
|
/**/TEMP/
|
||||||
|
/**/packages/
|
||||||
|
/**/bin/
|
||||||
|
/**/obj/
|
||||||
|
_site
|
||||||
|
_DocfxTemplate
|
||||||
5
docs/api/.gitignore
vendored
Normal file
5
docs/api/.gitignore
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
###############
|
||||||
|
# temp file #
|
||||||
|
###############
|
||||||
|
*.yml
|
||||||
|
.manifest
|
||||||
70
docs/docfx.json
Normal file
70
docs/docfx.json
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
{
|
||||||
|
"metadata": [
|
||||||
|
{
|
||||||
|
"src": [
|
||||||
|
{
|
||||||
|
"files": [
|
||||||
|
"UniTask/Assets/Plugins/UniTask/Runtime/**/*.cs"
|
||||||
|
],
|
||||||
|
"src": "../src"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"dest": "api",
|
||||||
|
"disableGitFeatures": false,
|
||||||
|
"disableDefaultFilter": false
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"build": {
|
||||||
|
"globalMetadata": {
|
||||||
|
"_disableContribution": true,
|
||||||
|
"_appTitle": "UniTask"
|
||||||
|
},
|
||||||
|
"content": [
|
||||||
|
{
|
||||||
|
"files": [
|
||||||
|
"api/**.yml",
|
||||||
|
"api/index.md"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"files": [
|
||||||
|
"articles/**.md",
|
||||||
|
"articles/**/toc.yml",
|
||||||
|
"toc.yml",
|
||||||
|
"*.md"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"resource": [
|
||||||
|
{
|
||||||
|
"files": [
|
||||||
|
"images/**"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"overwrite": [
|
||||||
|
{
|
||||||
|
"files": [
|
||||||
|
"apidoc/**.md"
|
||||||
|
],
|
||||||
|
"exclude": [
|
||||||
|
"obj/**",
|
||||||
|
"_site/**"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"dest": "_site",
|
||||||
|
|
||||||
|
"globalMetadataFiles": [],
|
||||||
|
"fileMetadataFiles": [],
|
||||||
|
"template": [
|
||||||
|
"_DocfxTemplate/templates/default-v2.5.2",
|
||||||
|
"_DocfxTemplate/templates/cysharp"
|
||||||
|
],
|
||||||
|
"postProcessors": [],
|
||||||
|
"markdownEngineName": "markdig",
|
||||||
|
"noLangKeyword": false,
|
||||||
|
"keepFileLink": false,
|
||||||
|
"cleanupCacheHistory": false
|
||||||
|
}
|
||||||
|
}
|
||||||
8
docs/index.md
Normal file
8
docs/index.md
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
---
|
||||||
|
title: Home
|
||||||
|
---
|
||||||
|
# UniTask
|
||||||
|
|
||||||
|
Provides an efficient async/await integration to Unity.
|
||||||
|
|
||||||
|
https://github.com/Cysharp/UniTask
|
||||||
11
docs/toc.yml
Normal file
11
docs/toc.yml
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
- name: API Documentation
|
||||||
|
href: api/
|
||||||
|
homepage: api/Cysharp.Threading.Tasks.html
|
||||||
|
|
||||||
|
- name: Repository
|
||||||
|
href: https://github.com/Cysharp/UniTask
|
||||||
|
homepage: https://github.com/Cysharp/UniTask
|
||||||
|
|
||||||
|
- name: Releases
|
||||||
|
href: https://github.com/Cysharp/UniTask/releases
|
||||||
|
homepage: https://github.com/Cysharp/UniTask/releases
|
||||||
BIN
src/UniTask.NetCore/Icon.png
Normal file
BIN
src/UniTask.NetCore/Icon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 3.1 KiB |
@@ -1,7 +1,6 @@
|
|||||||
#pragma warning disable 0649
|
#pragma warning disable 0649
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using System.Threading.Tasks.Sources;
|
using System.Threading.Tasks.Sources;
|
||||||
|
|
||||||
@@ -9,115 +8,91 @@ namespace Cysharp.Threading.Tasks
|
|||||||
{
|
{
|
||||||
public static class UniTaskValueTaskExtensions
|
public static class UniTaskValueTaskExtensions
|
||||||
{
|
{
|
||||||
public static ValueTask AsValueTask(this UniTask task)
|
public static ValueTask AsValueTask(this in UniTask task)
|
||||||
{
|
{
|
||||||
ref var core = ref Unsafe.As<UniTask, UniTaskToValueTask>(ref task);
|
#if NETSTANDARD2_0
|
||||||
if (core.source == null)
|
return new ValueTask(new UniTaskValueTaskSource(task), 0);
|
||||||
{
|
#else
|
||||||
return default;
|
return task;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
return new ValueTask(new UniTaskValueTaskSource(core.source), core.token);
|
public static ValueTask<T> AsValueTask<T>(this in UniTask<T> task)
|
||||||
|
{
|
||||||
|
#if NETSTANDARD2_0
|
||||||
|
return new ValueTask<T>(new UniTaskValueTaskSource<T>(task), 0);
|
||||||
|
#else
|
||||||
|
return task;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
public static ValueTask<T> AsValueTask<T>(this UniTask<T> task)
|
public static UniTask<T> AsUniTask<T>(this ValueTask<T> task, bool useCurrentSynchronizationContext = true)
|
||||||
{
|
{
|
||||||
ref var core = ref Unsafe.As<UniTask<T>, UniTaskToValueTask<T>>(ref task);
|
// NOTE: get _obj and _token directly for low overhead conversion but not yet implemented.
|
||||||
if (core.source == null)
|
return task.AsTask().AsUniTask(useCurrentSynchronizationContext);
|
||||||
{
|
|
||||||
return new ValueTask<T>(core.result);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return new ValueTask<T>(new UniTaskValueTaskSource<T>(core.source), core.token);
|
public static UniTask AsUniTask(this ValueTask task, bool useCurrentSynchronizationContext = true)
|
||||||
|
{
|
||||||
|
return task.AsTask().AsUniTask(useCurrentSynchronizationContext);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct UniTaskToValueTask
|
#if NETSTANDARD2_0
|
||||||
{
|
|
||||||
public IUniTaskSource source;
|
|
||||||
public short token;
|
|
||||||
}
|
|
||||||
|
|
||||||
class UniTaskValueTaskSource : IValueTaskSource
|
class UniTaskValueTaskSource : IValueTaskSource
|
||||||
{
|
{
|
||||||
readonly IUniTaskSource source;
|
readonly UniTask task;
|
||||||
|
readonly UniTask.Awaiter awaiter;
|
||||||
|
|
||||||
public UniTaskValueTaskSource(IUniTaskSource source)
|
public UniTaskValueTaskSource(UniTask task)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.task = task;
|
||||||
|
this.awaiter = task.GetAwaiter();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void GetResult(short token)
|
public void GetResult(short token)
|
||||||
{
|
{
|
||||||
source.GetResult(token);
|
awaiter.GetResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
public ValueTaskSourceStatus GetStatus(short token)
|
public ValueTaskSourceStatus GetStatus(short token)
|
||||||
{
|
{
|
||||||
var status = source.GetStatus(token);
|
return (ValueTaskSourceStatus)task.Status;
|
||||||
switch (status)
|
|
||||||
{
|
|
||||||
case UniTaskStatus.Pending:
|
|
||||||
return ValueTaskSourceStatus.Pending;
|
|
||||||
case UniTaskStatus.Succeeded:
|
|
||||||
return ValueTaskSourceStatus.Succeeded;
|
|
||||||
case UniTaskStatus.Faulted:
|
|
||||||
return ValueTaskSourceStatus.Faulted;
|
|
||||||
case UniTaskStatus.Canceled:
|
|
||||||
return ValueTaskSourceStatus.Canceled;
|
|
||||||
default:
|
|
||||||
return (ValueTaskSourceStatus)status;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void OnCompleted(Action<object> continuation, object state, short token, ValueTaskSourceOnCompletedFlags flags)
|
public void OnCompleted(Action<object> continuation, object state, short token, ValueTaskSourceOnCompletedFlags flags)
|
||||||
{
|
{
|
||||||
source.OnCompleted(continuation, state, token);
|
awaiter.SourceOnCompleted(continuation, state);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct UniTaskToValueTask<T>
|
|
||||||
{
|
|
||||||
public IUniTaskSource<T> source;
|
|
||||||
public T result;
|
|
||||||
public short token;
|
|
||||||
}
|
|
||||||
|
|
||||||
class UniTaskValueTaskSource<T> : IValueTaskSource<T>
|
class UniTaskValueTaskSource<T> : IValueTaskSource<T>
|
||||||
{
|
{
|
||||||
readonly IUniTaskSource<T> source;
|
readonly UniTask<T> task;
|
||||||
|
readonly UniTask<T>.Awaiter awaiter;
|
||||||
|
|
||||||
public UniTaskValueTaskSource(IUniTaskSource<T> source)
|
public UniTaskValueTaskSource(UniTask<T> task)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.task = task;
|
||||||
|
this.awaiter = task.GetAwaiter();
|
||||||
}
|
}
|
||||||
|
|
||||||
public T GetResult(short token)
|
public T GetResult(short token)
|
||||||
{
|
{
|
||||||
return source.GetResult(token);
|
return awaiter.GetResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
public ValueTaskSourceStatus GetStatus(short token)
|
public ValueTaskSourceStatus GetStatus(short token)
|
||||||
{
|
{
|
||||||
var status = source.GetStatus(token);
|
return (ValueTaskSourceStatus)task.Status;
|
||||||
switch (status)
|
|
||||||
{
|
|
||||||
case UniTaskStatus.Pending:
|
|
||||||
return ValueTaskSourceStatus.Pending;
|
|
||||||
case UniTaskStatus.Succeeded:
|
|
||||||
return ValueTaskSourceStatus.Succeeded;
|
|
||||||
case UniTaskStatus.Faulted:
|
|
||||||
return ValueTaskSourceStatus.Faulted;
|
|
||||||
case UniTaskStatus.Canceled:
|
|
||||||
return ValueTaskSourceStatus.Canceled;
|
|
||||||
default:
|
|
||||||
return (ValueTaskSourceStatus)status;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void OnCompleted(Action<object> continuation, object state, short token, ValueTaskSourceOnCompletedFlags flags)
|
public void OnCompleted(Action<object> continuation, object state, short token, ValueTaskSourceOnCompletedFlags flags)
|
||||||
{
|
{
|
||||||
source.OnCompleted(continuation, state, token);
|
awaiter.SourceOnCompleted(continuation, state);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
39
src/UniTask.NetCore/NetCore/UniTask.Delay.cs
Normal file
39
src/UniTask.NetCore/NetCore/UniTask.Delay.cs
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
//using Cysharp.Threading.Tasks.Internal;
|
||||||
|
//using System;
|
||||||
|
//using System.Collections.Concurrent;
|
||||||
|
//using System.Runtime.CompilerServices;
|
||||||
|
//using System.Threading;
|
||||||
|
|
||||||
|
//namespace Cysharp.Threading.Tasks
|
||||||
|
//{
|
||||||
|
// public partial struct UniTask
|
||||||
|
// {
|
||||||
|
// public static UniTask Delay()
|
||||||
|
// {
|
||||||
|
// return default;
|
||||||
|
// }
|
||||||
|
|
||||||
|
// sealed class DelayPromise : IUniTaskSource
|
||||||
|
// {
|
||||||
|
// public void GetResult(short token)
|
||||||
|
// {
|
||||||
|
// throw new NotImplementedException();
|
||||||
|
// }
|
||||||
|
|
||||||
|
// public UniTaskStatus GetStatus(short token)
|
||||||
|
// {
|
||||||
|
// throw new NotImplementedException();
|
||||||
|
// }
|
||||||
|
|
||||||
|
// public void OnCompleted(Action<object> continuation, object state, short token)
|
||||||
|
// {
|
||||||
|
// throw new NotImplementedException();
|
||||||
|
// }
|
||||||
|
|
||||||
|
// public UniTaskStatus UnsafeGetStatus()
|
||||||
|
// {
|
||||||
|
// throw new NotImplementedException();
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//}
|
||||||
@@ -1,4 +1,6 @@
|
|||||||
using System;
|
using Cysharp.Threading.Tasks.Internal;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Concurrent;
|
||||||
using System.Runtime.CompilerServices;
|
using System.Runtime.CompilerServices;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
|
|
||||||
@@ -41,7 +43,11 @@ namespace Cysharp.Threading.Tasks
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
#if NETCOREAPP3_1
|
||||||
|
ThreadPool.UnsafeQueueUserWorkItem(ThreadPoolWorkItem.Create(continuation), false);
|
||||||
|
#else
|
||||||
ThreadPool.UnsafeQueueUserWorkItem(WaitCallbackDelegate, continuation);
|
ThreadPool.UnsafeQueueUserWorkItem(WaitCallbackDelegate, continuation);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -50,6 +56,47 @@ namespace Cysharp.Threading.Tasks
|
|||||||
((Action)state).Invoke();
|
((Action)state).Invoke();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if NETCOREAPP3_1
|
||||||
|
|
||||||
|
sealed class ThreadPoolWorkItem : IThreadPoolWorkItem, ITaskPoolNode<ThreadPoolWorkItem>
|
||||||
|
{
|
||||||
|
static TaskPool<ThreadPoolWorkItem> pool;
|
||||||
|
public ThreadPoolWorkItem NextNode { get; set; }
|
||||||
|
|
||||||
|
static ThreadPoolWorkItem()
|
||||||
|
{
|
||||||
|
TaskPool.RegisterSizeGetter(typeof(ThreadPoolWorkItem), () => pool.Size);
|
||||||
|
}
|
||||||
|
|
||||||
|
Action continuation;
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public static ThreadPoolWorkItem Create(Action continuation)
|
||||||
|
{
|
||||||
|
if (!pool.TryPop(out var item))
|
||||||
|
{
|
||||||
|
item = new ThreadPoolWorkItem();
|
||||||
|
}
|
||||||
|
|
||||||
|
item.continuation = continuation;
|
||||||
|
return item;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public void Execute()
|
||||||
|
{
|
||||||
|
var call = continuation;
|
||||||
|
continuation = null;
|
||||||
|
if (call != null)
|
||||||
|
{
|
||||||
|
pool.TryPush(this);
|
||||||
|
call.Invoke();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,11 +1,30 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk">
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<TargetFramework>netstandard2.0</TargetFramework>
|
<TargetFrameworks>netcoreapp3.1;netstandard2.1;netstandard2.0</TargetFrameworks>
|
||||||
<AssemblyName>UniTask</AssemblyName>
|
<AssemblyName>UniTask</AssemblyName>
|
||||||
|
<LangVersion>8.0</LangVersion>
|
||||||
<RootNamespace>Cysharp.Threading.Tasks</RootNamespace>
|
<RootNamespace>Cysharp.Threading.Tasks</RootNamespace>
|
||||||
|
|
||||||
|
<!-- NuGet Packaging -->
|
||||||
|
<Id>UniTask</Id>
|
||||||
|
<PackageVersion>$(Version)</PackageVersion>
|
||||||
|
<Company>Cysharp</Company>
|
||||||
|
<Authors>Cysharp</Authors>
|
||||||
|
<Copyright>© Cysharp, Inc.</Copyright>
|
||||||
|
<PackageTags>task;async</PackageTags>
|
||||||
|
<Description>Provides an efficient async/await integration to Unity and .NET Core.</Description>
|
||||||
|
<PackageProjectUrl>https://github.com/Cysharp/UniTask</PackageProjectUrl>
|
||||||
|
<RepositoryUrl>$(PackageProjectUrl)</RepositoryUrl>
|
||||||
|
<RepositoryType>git</RepositoryType>
|
||||||
|
<PackageLicenseExpression>MIT</PackageLicenseExpression>
|
||||||
|
<PackageIcon>Icon.png</PackageIcon>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<None Include="Icon.png" Pack="true" PackagePath="/" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Compile Include="..\UniTask\Assets\Plugins\UniTask\Runtime\**\*.cs"
|
<Compile Include="..\UniTask\Assets\Plugins\UniTask\Runtime\**\*.cs"
|
||||||
Exclude="
|
Exclude="
|
||||||
@@ -16,6 +35,7 @@
|
|||||||
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\UnityEqualityComparer.cs;
|
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\UnityEqualityComparer.cs;
|
||||||
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\DiagnosticsExtensions.cs;
|
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\DiagnosticsExtensions.cs;
|
||||||
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\PlayerLoopRunner.cs;
|
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\PlayerLoopRunner.cs;
|
||||||
|
..\UniTask\Assets\Plugins\UniTask\Runtime\Internal\ContinuationQueue.cs;
|
||||||
|
|
||||||
..\UniTask\Assets\Plugins\UniTask\Runtime\CancellationTokenSourceExtensions.cs;
|
..\UniTask\Assets\Plugins\UniTask\Runtime\CancellationTokenSourceExtensions.cs;
|
||||||
..\UniTask\Assets\Plugins\UniTask\Runtime\EnumeratorAsyncExtensions.cs;
|
..\UniTask\Assets\Plugins\UniTask\Runtime\EnumeratorAsyncExtensions.cs;
|
||||||
@@ -27,6 +47,7 @@
|
|||||||
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityAsyncExtensions.cs;
|
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityAsyncExtensions.cs;
|
||||||
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityAsyncExtensions.uGUI.cs;
|
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityAsyncExtensions.uGUI.cs;
|
||||||
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityAsyncExtensions.MonoBehaviour.cs;
|
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityAsyncExtensions.MonoBehaviour.cs;
|
||||||
|
..\UniTask\Assets\Plugins\UniTask\Runtime\UnityBindingExtensions.cs;
|
||||||
" />
|
" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
|
|||||||
258
src/UniTask.NetCoreSandbox/AllocationCheck.cs
Normal file
258
src/UniTask.NetCoreSandbox/AllocationCheck.cs
Normal file
@@ -0,0 +1,258 @@
|
|||||||
|
using BenchmarkDotNet.Attributes;
|
||||||
|
using System.Linq;
|
||||||
|
using BenchmarkDotNet.Configs;
|
||||||
|
using BenchmarkDotNet.Diagnosers;
|
||||||
|
using BenchmarkDotNet.Exporters;
|
||||||
|
using BenchmarkDotNet.Jobs;
|
||||||
|
using BenchmarkDotNet.Running;
|
||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using PooledAwait;
|
||||||
|
using System;
|
||||||
|
using System.Runtime.ExceptionServices;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using Cysharp.Threading.Tasks.CompilerServices;
|
||||||
|
using System.Collections.Concurrent;
|
||||||
|
|
||||||
|
[Config(typeof(BenchmarkConfig))]
|
||||||
|
public class AllocationCheck
|
||||||
|
{
|
||||||
|
// note: all the benchmarks use Task/Task<T> for the public API, because BenchmarkDotNet
|
||||||
|
// doesn't work reliably with more exotic task-types (even just ValueTask fails); instead,
|
||||||
|
// we'll obscure the cost of the outer awaitable by doing a relatively large number of
|
||||||
|
// iterations, so that we're only really measuring the inner loop
|
||||||
|
private const int InnerOps = 1000;
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps)]
|
||||||
|
public async Task ViaUniTask()
|
||||||
|
{
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
{
|
||||||
|
var a = Core();
|
||||||
|
var b = Core();
|
||||||
|
var c = Core();
|
||||||
|
await a;
|
||||||
|
await b;
|
||||||
|
await c;
|
||||||
|
}
|
||||||
|
|
||||||
|
static async UniTask Core()
|
||||||
|
{
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps)]
|
||||||
|
public async Task<int> ViaUniTaskT()
|
||||||
|
{
|
||||||
|
var sum = 0;
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
{
|
||||||
|
var a = Core();
|
||||||
|
var b = Core();
|
||||||
|
var c = Core();
|
||||||
|
sum += await a;
|
||||||
|
sum += await b;
|
||||||
|
sum += await c;
|
||||||
|
}
|
||||||
|
return sum;
|
||||||
|
|
||||||
|
static async UniTask<int> Core()
|
||||||
|
{
|
||||||
|
var a = await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 10);
|
||||||
|
var b = await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 10);
|
||||||
|
var c = await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 10);
|
||||||
|
return 10;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//[Benchmark(OperationsPerInvoke = InnerOps)]
|
||||||
|
//[Benchmark]
|
||||||
|
public void ViaUniTaskVoid()
|
||||||
|
{
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
{
|
||||||
|
Core().Forget();
|
||||||
|
Core().Forget();
|
||||||
|
Core().Forget();
|
||||||
|
}
|
||||||
|
|
||||||
|
static async UniTaskVoid Core()
|
||||||
|
{
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Foo : IAsyncStateMachine
|
||||||
|
{
|
||||||
|
public AsyncUniTaskVoidMethodBuilder builder;
|
||||||
|
public TestAwaiter awaiter;
|
||||||
|
public TestAwaiter awaiterawaiter;
|
||||||
|
|
||||||
|
public int state;
|
||||||
|
|
||||||
|
public void MoveNext()
|
||||||
|
{
|
||||||
|
switch (state)
|
||||||
|
{
|
||||||
|
case -1:
|
||||||
|
awaiterawaiter = awaiter.GetAwaiter();
|
||||||
|
if (awaiterawaiter.IsCompleted)
|
||||||
|
{
|
||||||
|
goto case 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
state = 0;
|
||||||
|
builder.AwaitUnsafeOnCompleted(ref awaiterawaiter, ref this);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0:
|
||||||
|
default:
|
||||||
|
goto END;
|
||||||
|
}
|
||||||
|
|
||||||
|
END:
|
||||||
|
builder.SetResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetStateMachine(IAsyncStateMachine stateMachine)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class TaskTestException : Exception
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct TestAwaiter : ICriticalNotifyCompletion
|
||||||
|
{
|
||||||
|
readonly UniTaskStatus status;
|
||||||
|
readonly bool isCompleted;
|
||||||
|
|
||||||
|
public TestAwaiter(bool isCompleted, UniTaskStatus status)
|
||||||
|
{
|
||||||
|
this.isCompleted = isCompleted;
|
||||||
|
this.status = status;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TestAwaiter GetAwaiter() => this;
|
||||||
|
|
||||||
|
public bool IsCompleted => isCompleted;
|
||||||
|
|
||||||
|
public void GetResult()
|
||||||
|
{
|
||||||
|
switch (status)
|
||||||
|
{
|
||||||
|
case UniTaskStatus.Faulted:
|
||||||
|
throw new TaskTestException();
|
||||||
|
case UniTaskStatus.Canceled:
|
||||||
|
throw new OperationCanceledException();
|
||||||
|
case UniTaskStatus.Pending:
|
||||||
|
case UniTaskStatus.Succeeded:
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.UnsafeQueueUserWorkItem(ThreadPoolWorkItem.Create(continuation), false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.UnsafeQueueUserWorkItem(ThreadPoolWorkItem.Create(continuation), false);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct TestAwaiter<T> : ICriticalNotifyCompletion
|
||||||
|
{
|
||||||
|
readonly UniTaskStatus status;
|
||||||
|
readonly bool isCompleted;
|
||||||
|
readonly T value;
|
||||||
|
|
||||||
|
public TestAwaiter(bool isCompleted, UniTaskStatus status, T value)
|
||||||
|
{
|
||||||
|
this.isCompleted = isCompleted;
|
||||||
|
this.status = status;
|
||||||
|
this.value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TestAwaiter<T> GetAwaiter() => this;
|
||||||
|
|
||||||
|
public bool IsCompleted => isCompleted;
|
||||||
|
|
||||||
|
public T GetResult()
|
||||||
|
{
|
||||||
|
switch (status)
|
||||||
|
{
|
||||||
|
case UniTaskStatus.Faulted:
|
||||||
|
throw new TaskTestException();
|
||||||
|
case UniTaskStatus.Canceled:
|
||||||
|
throw new OperationCanceledException();
|
||||||
|
case UniTaskStatus.Pending:
|
||||||
|
case UniTaskStatus.Succeeded:
|
||||||
|
default:
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.UnsafeQueueUserWorkItem(ThreadPoolWorkItem.Create(continuation), false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.UnsafeQueueUserWorkItem(ThreadPoolWorkItem.Create(continuation), false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public sealed class ThreadPoolWorkItem : IThreadPoolWorkItem
|
||||||
|
{
|
||||||
|
public static readonly ConcurrentQueue<ThreadPoolWorkItem> pool = new ConcurrentQueue<ThreadPoolWorkItem>();
|
||||||
|
|
||||||
|
public static void CreatePoolItems(int count)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < count; i++)
|
||||||
|
{
|
||||||
|
pool.Enqueue(new ThreadPoolWorkItem());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Action continuation;
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public static ThreadPoolWorkItem Create(Action continuation)
|
||||||
|
{
|
||||||
|
if (!pool.TryDequeue(out var item))
|
||||||
|
{
|
||||||
|
item = new ThreadPoolWorkItem();
|
||||||
|
}
|
||||||
|
|
||||||
|
item.continuation = continuation;
|
||||||
|
return item;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public void Execute()
|
||||||
|
{
|
||||||
|
var call = continuation;
|
||||||
|
continuation = null;
|
||||||
|
pool.Enqueue(this);
|
||||||
|
|
||||||
|
call.Invoke();
|
||||||
|
}
|
||||||
|
}
|
||||||
283
src/UniTask.NetCoreSandbox/Benchmark.cs
Normal file
283
src/UniTask.NetCoreSandbox/Benchmark.cs
Normal file
@@ -0,0 +1,283 @@
|
|||||||
|
using BenchmarkDotNet.Attributes;
|
||||||
|
using System.Linq;
|
||||||
|
using BenchmarkDotNet.Configs;
|
||||||
|
using BenchmarkDotNet.Diagnosers;
|
||||||
|
using BenchmarkDotNet.Exporters;
|
||||||
|
using BenchmarkDotNet.Jobs;
|
||||||
|
using BenchmarkDotNet.Running;
|
||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using PooledAwait;
|
||||||
|
using System;
|
||||||
|
using System.Runtime.ExceptionServices;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using Cysharp.Threading.Tasks.CompilerServices;
|
||||||
|
|
||||||
|
//class Program
|
||||||
|
//{
|
||||||
|
// static void Main(string[] args)
|
||||||
|
// {
|
||||||
|
// var switcher = new BenchmarkSwitcher(new[]
|
||||||
|
// {
|
||||||
|
// typeof(StandardBenchmark)
|
||||||
|
// });
|
||||||
|
|
||||||
|
//#if DEBUG
|
||||||
|
// var b = new StandardBenchmark();
|
||||||
|
|
||||||
|
//#else
|
||||||
|
// switcher.Run(args);
|
||||||
|
//#endif
|
||||||
|
// }
|
||||||
|
//}
|
||||||
|
|
||||||
|
public class BenchmarkConfig : ManualConfig
|
||||||
|
{
|
||||||
|
public BenchmarkConfig()
|
||||||
|
{
|
||||||
|
AddDiagnoser(MemoryDiagnoser.Default);
|
||||||
|
AddJob(Job.ShortRun.WithLaunchCount(1).WithIterationCount(1).WithWarmupCount(1)/*.RunOncePerIteration()*/);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// borrowed from PooledAwait
|
||||||
|
|
||||||
|
[Config(typeof(BenchmarkConfig))]
|
||||||
|
[GroupBenchmarksBy(BenchmarkLogicalGroupRule.ByCategory)]
|
||||||
|
[CategoriesColumn]
|
||||||
|
public class ComparisonBenchmarks
|
||||||
|
{
|
||||||
|
// note: all the benchmarks use Task/Task<T> for the public API, because BenchmarkDotNet
|
||||||
|
// doesn't work reliably with more exotic task-types (even just ValueTask fails); instead,
|
||||||
|
// we'll obscure the cost of the outer awaitable by doing a relatively large number of
|
||||||
|
// iterations, so that we're only really measuring the inner loop
|
||||||
|
private const int InnerOps = 1000;
|
||||||
|
|
||||||
|
public bool ConfigureAwait { get; set; } = false;
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps, Description = ".NET")]
|
||||||
|
[BenchmarkCategory("Task<T>")]
|
||||||
|
public async Task<int> ViaTaskT()
|
||||||
|
{
|
||||||
|
int sum = 0;
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
sum += await Inner(1, 2).ConfigureAwait(ConfigureAwait);
|
||||||
|
return sum;
|
||||||
|
|
||||||
|
static async Task<int> Inner(int x, int y)
|
||||||
|
{
|
||||||
|
int i = x;
|
||||||
|
await Task.Yield();
|
||||||
|
i *= y;
|
||||||
|
await Task.Yield();
|
||||||
|
return 5 * i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps, Description = ".NET")]
|
||||||
|
[BenchmarkCategory("Task")]
|
||||||
|
public async Task ViaTask()
|
||||||
|
{
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
await Inner().ConfigureAwait(ConfigureAwait);
|
||||||
|
|
||||||
|
static async Task Inner()
|
||||||
|
{
|
||||||
|
await Task.Yield();
|
||||||
|
await Task.Yield();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps, Description = ".NET")]
|
||||||
|
[BenchmarkCategory("ValueTask<T>")]
|
||||||
|
public async Task<int> ViaValueTaskT()
|
||||||
|
{
|
||||||
|
int sum = 0;
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
sum += await Inner(1, 2).ConfigureAwait(ConfigureAwait);
|
||||||
|
return sum;
|
||||||
|
|
||||||
|
static async ValueTask<int> Inner(int x, int y)
|
||||||
|
{
|
||||||
|
int i = x;
|
||||||
|
await Task.Yield();
|
||||||
|
i *= y;
|
||||||
|
await Task.Yield();
|
||||||
|
return 5 * i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps, Description = ".NET")]
|
||||||
|
[BenchmarkCategory("ValueTask")]
|
||||||
|
public async Task ViaValueTask()
|
||||||
|
{
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
await Inner().ConfigureAwait(ConfigureAwait);
|
||||||
|
|
||||||
|
static async ValueTask Inner()
|
||||||
|
{
|
||||||
|
await Task.Yield();
|
||||||
|
await Task.Yield();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps, Description = "Pooled")]
|
||||||
|
[BenchmarkCategory("ValueTask<T>")]
|
||||||
|
public async Task<int> ViaPooledValueTaskT()
|
||||||
|
{
|
||||||
|
int sum = 0;
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
sum += await Inner(1, 2).ConfigureAwait(ConfigureAwait);
|
||||||
|
return sum;
|
||||||
|
|
||||||
|
static async PooledValueTask<int> Inner(int x, int y)
|
||||||
|
{
|
||||||
|
int i = x;
|
||||||
|
await Task.Yield();
|
||||||
|
i *= y;
|
||||||
|
await Task.Yield();
|
||||||
|
return 5 * i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps, Description = "Pooled")]
|
||||||
|
[BenchmarkCategory("ValueTask")]
|
||||||
|
public async Task ViaPooledValueTask()
|
||||||
|
{
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
await Inner().ConfigureAwait(ConfigureAwait);
|
||||||
|
|
||||||
|
static async PooledValueTask Inner()
|
||||||
|
{
|
||||||
|
await Task.Yield();
|
||||||
|
await Task.Yield();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps, Description = "Pooled")]
|
||||||
|
[BenchmarkCategory("Task<T>")]
|
||||||
|
public async Task<int> ViaPooledTaskT()
|
||||||
|
{
|
||||||
|
int sum = 0;
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
sum += await Inner(1, 2).ConfigureAwait(ConfigureAwait);
|
||||||
|
return sum;
|
||||||
|
|
||||||
|
static async PooledTask<int> Inner(int x, int y)
|
||||||
|
{
|
||||||
|
int i = x;
|
||||||
|
await Task.Yield();
|
||||||
|
i *= y;
|
||||||
|
await Task.Yield();
|
||||||
|
return 5 * i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps, Description = "Pooled")]
|
||||||
|
[BenchmarkCategory("Task")]
|
||||||
|
public async Task ViaPooledTask()
|
||||||
|
{
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
await Inner().ConfigureAwait(ConfigureAwait);
|
||||||
|
|
||||||
|
static async PooledTask Inner()
|
||||||
|
{
|
||||||
|
await Task.Yield();
|
||||||
|
await Task.Yield();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ---
|
||||||
|
|
||||||
|
//[Benchmark(OperationsPerInvoke = InnerOps, Description = "UniTaskVoid")]
|
||||||
|
//[BenchmarkCategory("UniTask")]
|
||||||
|
//public async Task ViaUniTaskVoid()
|
||||||
|
//{
|
||||||
|
// for (int i = 0; i < InnerOps; i++)
|
||||||
|
// {
|
||||||
|
// await Inner();
|
||||||
|
// }
|
||||||
|
|
||||||
|
// static async UniTaskVoid Inner()
|
||||||
|
// {
|
||||||
|
// await UniTask.Yield();
|
||||||
|
// await UniTask.Yield();
|
||||||
|
// }
|
||||||
|
//}
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps, Description = "UniTask")]
|
||||||
|
[BenchmarkCategory("UniTask")]
|
||||||
|
public async Task ViaUniTask()
|
||||||
|
{
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
{
|
||||||
|
await Inner();
|
||||||
|
}
|
||||||
|
|
||||||
|
static async UniTask Inner()
|
||||||
|
{
|
||||||
|
await UniTask.Yield();
|
||||||
|
await UniTask.Yield();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark(OperationsPerInvoke = InnerOps, Description = "UniTaskT")]
|
||||||
|
[BenchmarkCategory("UniTask")]
|
||||||
|
public async Task<int> ViaUniTaskT()
|
||||||
|
{
|
||||||
|
var sum = 0;
|
||||||
|
for (int i = 0; i < InnerOps; i++)
|
||||||
|
{
|
||||||
|
sum += await Inner(1, 2);
|
||||||
|
}
|
||||||
|
return sum;
|
||||||
|
|
||||||
|
static async UniTask<int> Inner(int x, int y)
|
||||||
|
{
|
||||||
|
int i = x;
|
||||||
|
await UniTask.Yield();
|
||||||
|
i *= y;
|
||||||
|
await UniTask.Yield();
|
||||||
|
return 5 * i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct MyAwaiter : ICriticalNotifyCompletion
|
||||||
|
{
|
||||||
|
public MyAwaiter GetAwaiter() => this;
|
||||||
|
|
||||||
|
public bool IsCompleted => false;
|
||||||
|
|
||||||
|
public void GetResult()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
continuation();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
continuation();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct MyTestStateMachine : IAsyncStateMachine
|
||||||
|
{
|
||||||
|
public void MoveNext()
|
||||||
|
{
|
||||||
|
//throw new NotImplementedException();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetStateMachine(IAsyncStateMachine stateMachine)
|
||||||
|
{
|
||||||
|
//throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -15,6 +15,175 @@ using System.Reactive.Concurrency;
|
|||||||
|
|
||||||
namespace NetCoreSandbox
|
namespace NetCoreSandbox
|
||||||
{
|
{
|
||||||
|
public class MySyncContext : SynchronizationContext
|
||||||
|
{
|
||||||
|
public MySyncContext()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void Post(SendOrPostCallback d, object state)
|
||||||
|
{
|
||||||
|
Console.WriteLine("Called SyncContext Post!");
|
||||||
|
base.Post(d, state);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class Text
|
||||||
|
{
|
||||||
|
|
||||||
|
public string text { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public class ZeroAllocAsyncAwaitInDotNetCore
|
||||||
|
{
|
||||||
|
public ValueTask<int> NanikaAsync(int x, int y)
|
||||||
|
{
|
||||||
|
return Core(this, x, y);
|
||||||
|
|
||||||
|
static async UniTask<int> Core(ZeroAllocAsyncAwaitInDotNetCore self, int x, int y)
|
||||||
|
{
|
||||||
|
// nanika suru...
|
||||||
|
await Task.Delay(TimeSpan.FromSeconds(x + y));
|
||||||
|
|
||||||
|
return 10;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public class TaskTestException : Exception
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public struct TestAwaiter : ICriticalNotifyCompletion
|
||||||
|
{
|
||||||
|
readonly UniTaskStatus status;
|
||||||
|
readonly bool isCompleted;
|
||||||
|
|
||||||
|
public TestAwaiter(bool isCompleted, UniTaskStatus status)
|
||||||
|
{
|
||||||
|
this.isCompleted = isCompleted;
|
||||||
|
this.status = status;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TestAwaiter GetAwaiter() => this;
|
||||||
|
|
||||||
|
public bool IsCompleted => isCompleted;
|
||||||
|
|
||||||
|
public void GetResult()
|
||||||
|
{
|
||||||
|
switch (status)
|
||||||
|
{
|
||||||
|
case UniTaskStatus.Faulted:
|
||||||
|
throw new TaskTestException();
|
||||||
|
case UniTaskStatus.Canceled:
|
||||||
|
throw new OperationCanceledException();
|
||||||
|
case UniTaskStatus.Pending:
|
||||||
|
case UniTaskStatus.Succeeded:
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.QueueUserWorkItem(_ => continuation(), null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.UnsafeQueueUserWorkItem(_ => continuation(), null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public struct TestAwaiter<T> : ICriticalNotifyCompletion
|
||||||
|
{
|
||||||
|
readonly UniTaskStatus status;
|
||||||
|
readonly bool isCompleted;
|
||||||
|
readonly T value;
|
||||||
|
|
||||||
|
public TestAwaiter(bool isCompleted, UniTaskStatus status, T value)
|
||||||
|
{
|
||||||
|
this.isCompleted = isCompleted;
|
||||||
|
this.status = status;
|
||||||
|
this.value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TestAwaiter<T> GetAwaiter() => this;
|
||||||
|
|
||||||
|
public bool IsCompleted => isCompleted;
|
||||||
|
|
||||||
|
public T GetResult()
|
||||||
|
{
|
||||||
|
switch (status)
|
||||||
|
{
|
||||||
|
case UniTaskStatus.Faulted:
|
||||||
|
throw new TaskTestException();
|
||||||
|
case UniTaskStatus.Canceled:
|
||||||
|
throw new OperationCanceledException();
|
||||||
|
case UniTaskStatus.Pending:
|
||||||
|
case UniTaskStatus.Succeeded:
|
||||||
|
default:
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.QueueUserWorkItem(_ => continuation(), null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.UnsafeQueueUserWorkItem(_ => continuation(), null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static partial class UnityUIComponentExtensions
|
||||||
|
{
|
||||||
|
public static void BindTo(this IUniTaskAsyncEnumerable<string> source, Text text)
|
||||||
|
{
|
||||||
|
AAAACORECORE(source, text).Forget();
|
||||||
|
|
||||||
|
async UniTaskVoid AAAACORECORE(IUniTaskAsyncEnumerable<string> source2, Text text2)
|
||||||
|
{
|
||||||
|
var e = source2.GetAsyncEnumerator();
|
||||||
|
try
|
||||||
|
{
|
||||||
|
while (await e.MoveNextAsync())
|
||||||
|
{
|
||||||
|
text2.text = e.Current;
|
||||||
|
// action(e.Current);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
if (e != null)
|
||||||
|
{
|
||||||
|
await e.DisposeAsync();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//public static IDisposable SubscribeToText<T>(this IObservable<T> source, Text text)
|
||||||
|
//{
|
||||||
|
// return source.SubscribeWithState(text, (x, t) => t.text = x.ToString());
|
||||||
|
//}
|
||||||
|
|
||||||
|
//public static IDisposable SubscribeToText<T>(this IObservable<T> source, Text text, Func<T, string> selector)
|
||||||
|
//{
|
||||||
|
// return source.SubscribeWithState2(text, selector, (x, t, s) => t.text = s(x));
|
||||||
|
//}
|
||||||
|
|
||||||
|
//public static IDisposable SubscribeToInteractable(this IObservable<bool> source, Selectable selectable)
|
||||||
|
//{
|
||||||
|
// return source.SubscribeWithState(selectable, (x, s) => s.interactable = x);
|
||||||
|
//}
|
||||||
|
}
|
||||||
|
|
||||||
class Program
|
class Program
|
||||||
{
|
{
|
||||||
static string FlattenGenArgs(Type type)
|
static string FlattenGenArgs(Type type)
|
||||||
@@ -37,33 +206,144 @@ namespace NetCoreSandbox
|
|||||||
await Task.Delay(10, cancellationToken);
|
await Task.Delay(10, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
static async Task Main(string[] args)
|
public class MyDisposable : IDisposable
|
||||||
|
{
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void Test()
|
||||||
|
{
|
||||||
|
var disp = new MyDisposable();
|
||||||
|
|
||||||
|
using var _ = new MyDisposable();
|
||||||
|
|
||||||
|
Console.WriteLine("tako");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static async UniTask FooBarAsync()
|
||||||
|
{
|
||||||
|
await using (UniTask.ReturnToCurrentSynchronizationContext())
|
||||||
|
{
|
||||||
|
await UniTask.SwitchToThreadPool();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static async UniTask Aaa()
|
||||||
|
{
|
||||||
|
await FooBarAsync();
|
||||||
|
|
||||||
|
Console.WriteLine("FooBarAsync End");
|
||||||
|
}
|
||||||
|
|
||||||
|
static async UniTask WhereSelect()
|
||||||
{
|
{
|
||||||
await foreach (var item in UniTaskAsyncEnumerable.Range(1, 10)
|
await foreach (var item in UniTaskAsyncEnumerable.Range(1, 10)
|
||||||
.SelectAwait(x => UniTask.Run(() => x))
|
.SelectAwait(async x =>
|
||||||
.TakeLast(6)
|
{
|
||||||
|
await UniTask.Yield();
|
||||||
|
return x;
|
||||||
|
})
|
||||||
)
|
.Where(x => x % 2 == 0))
|
||||||
{
|
{
|
||||||
|
|
||||||
Console.WriteLine(item);
|
Console.WriteLine(item);
|
||||||
}
|
}
|
||||||
|
|
||||||
// AsyncEnumerable.Range(1,10).FirstAsync(
|
|
||||||
// AsyncEnumerable.Range(1, 10).GroupBy(x=>x).Select(x=>x.first
|
|
||||||
|
|
||||||
|
|
||||||
// AsyncEnumerable.Range(1,10).WithCancellation(CancellationToken.None).WithCancellation
|
|
||||||
|
|
||||||
|
|
||||||
//Enumerable.Range(1,10).ToHashSet(
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static async Task Main(string[] args)
|
||||||
|
{
|
||||||
|
#if !DEBUG
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//await new AllocationCheck().ViaUniTaskVoid();
|
||||||
|
//Console.ReadLine();
|
||||||
|
BenchmarkDotNet.Running.BenchmarkSwitcher.FromAssembly(typeof(Program).Assembly).Run(args);
|
||||||
|
|
||||||
|
//await new ComparisonBenchmarks().ViaUniTaskT();
|
||||||
|
return;
|
||||||
|
#endif
|
||||||
|
// await new AllocationCheck().ViaUniTaskVoid();
|
||||||
|
|
||||||
|
// AsyncTest().Forge
|
||||||
|
|
||||||
|
Console.WriteLine("A?");
|
||||||
|
var a = await new ZeroAllocAsyncAwaitInDotNetCore().NanikaAsync(1, 2);
|
||||||
|
Console.WriteLine("RET:" + a);
|
||||||
|
await WhereSelect();
|
||||||
|
|
||||||
|
SynchronizationContext.SetSynchronizationContext(new MySyncContext());
|
||||||
|
|
||||||
|
await Aaa();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//AsyncTest().Forget();
|
||||||
|
|
||||||
|
// AsyncTest().Forget();
|
||||||
|
|
||||||
|
ThreadPool.SetMinThreads(100, 100);
|
||||||
|
|
||||||
|
//List<UniTask<int>> list = new List<UniTask<int>>();
|
||||||
|
for (int i = 0; i < short.MaxValue; i++)
|
||||||
|
{
|
||||||
|
//// list.Add(AsyncTest());
|
||||||
|
await YieldCore();
|
||||||
|
}
|
||||||
|
//await UniTask.WhenAll(list);
|
||||||
|
|
||||||
|
//Console.WriteLine("TOGO");
|
||||||
|
|
||||||
|
//var a = await AsyncTest();
|
||||||
|
//var b = AsyncTest();
|
||||||
|
//var c = AsyncTest();
|
||||||
|
await YieldCore();
|
||||||
|
|
||||||
|
//await b;
|
||||||
|
//await c;
|
||||||
|
|
||||||
|
|
||||||
|
//foreach (var item in Cysharp.Threading.Tasks.Internal.TaskPool.GetCacheSizeInfo())
|
||||||
|
//{
|
||||||
|
// Console.WriteLine(item);
|
||||||
|
//}
|
||||||
|
|
||||||
|
Console.ReadLine();
|
||||||
|
}
|
||||||
|
|
||||||
|
static async UniTask YieldCore()
|
||||||
|
{
|
||||||
|
await UniTask.Yield();
|
||||||
|
}
|
||||||
|
|
||||||
|
#pragma warning disable CS1998
|
||||||
|
|
||||||
|
|
||||||
|
static async UniTask<int> AsyncTest()
|
||||||
|
{
|
||||||
|
// empty
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
await new TestAwaiter(true, UniTaskStatus.Succeeded);
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
return 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#pragma warning restore CS1998
|
||||||
|
|
||||||
void Foo()
|
void Foo()
|
||||||
{
|
{
|
||||||
|
|||||||
396
src/UniTask.NetCoreSandbox/QueueCheck.cs
Normal file
396
src/UniTask.NetCoreSandbox/QueueCheck.cs
Normal file
@@ -0,0 +1,396 @@
|
|||||||
|
using BenchmarkDotNet.Attributes;
|
||||||
|
using System.Collections.Concurrent;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
[Config(typeof(BenchmarkConfig))]
|
||||||
|
public class QueueCheck
|
||||||
|
{
|
||||||
|
Node node1 = new Node();
|
||||||
|
Node node2 = new Node();
|
||||||
|
Queue<Node> q1 = new Queue<Node>();
|
||||||
|
Stack<Node> s1 = new Stack<Node>();
|
||||||
|
ConcurrentQueue<Node> cq = new ConcurrentQueue<Node>();
|
||||||
|
ConcurrentStack<Node> cs = new ConcurrentStack<Node>();
|
||||||
|
static TaskPool<Node> pool;
|
||||||
|
static TaskPoolEqualNull<Node> poolEqualNull;
|
||||||
|
static TaskPoolClass<Node> poolClass = new TaskPoolClass<Node>();
|
||||||
|
static TaskPoolWithoutSize<Node> poolWithoutSize;
|
||||||
|
static TaskPoolWithoutLock<Node> poolWithoutLock;
|
||||||
|
|
||||||
|
[Benchmark]
|
||||||
|
public void Queue()
|
||||||
|
{
|
||||||
|
q1.Enqueue(node1);
|
||||||
|
q1.Enqueue(node1);
|
||||||
|
q1.TryDequeue(out _);
|
||||||
|
q1.TryDequeue(out _);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark]
|
||||||
|
public void QueueLock()
|
||||||
|
{
|
||||||
|
lock (q1) { q1.Enqueue(node1); }
|
||||||
|
lock (q1) { q1.Enqueue(node1); }
|
||||||
|
lock (q1) { q1.TryDequeue(out _); }
|
||||||
|
lock (q1) { q1.TryDequeue(out _); }
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark]
|
||||||
|
public void Stack()
|
||||||
|
{
|
||||||
|
s1.Push(node1);
|
||||||
|
s1.Push(node2);
|
||||||
|
s1.TryPop(out _);
|
||||||
|
s1.TryPop(out _);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark]
|
||||||
|
public void StackLock()
|
||||||
|
{
|
||||||
|
lock (s1) { s1.Push(node1); }
|
||||||
|
lock (s1) { s1.Push(node2); }
|
||||||
|
lock (s1) { s1.TryPop(out _); }
|
||||||
|
lock (s1) { s1.TryPop(out _); }
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark]
|
||||||
|
public void ConcurrentQueue()
|
||||||
|
{
|
||||||
|
cq.Enqueue(node1);
|
||||||
|
cq.Enqueue(node1);
|
||||||
|
cq.TryDequeue(out _);
|
||||||
|
cq.TryDequeue(out _);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark]
|
||||||
|
public void ConcurrentStack()
|
||||||
|
{
|
||||||
|
cs.Push(node1);
|
||||||
|
cs.Push(node2);
|
||||||
|
cs.TryPop(out _);
|
||||||
|
cs.TryPop(out _);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark]
|
||||||
|
public void TaskPool()
|
||||||
|
{
|
||||||
|
pool.TryPush(node1);
|
||||||
|
pool.TryPush(node2);
|
||||||
|
pool.TryPop(out _);
|
||||||
|
pool.TryPop(out _);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark]
|
||||||
|
public void TaskPoolEqualNull()
|
||||||
|
{
|
||||||
|
poolEqualNull.TryPush(node1);
|
||||||
|
poolEqualNull.TryPush(node2);
|
||||||
|
poolEqualNull.TryPop(out _);
|
||||||
|
poolEqualNull.TryPop(out _);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark]
|
||||||
|
public void TaskPoolClass()
|
||||||
|
{
|
||||||
|
poolClass.TryPush(node1);
|
||||||
|
poolClass.TryPush(node2);
|
||||||
|
poolClass.TryPop(out _);
|
||||||
|
poolClass.TryPop(out _);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark]
|
||||||
|
public void TaskPoolWithoutSize()
|
||||||
|
{
|
||||||
|
poolWithoutSize.TryPush(node1);
|
||||||
|
poolWithoutSize.TryPush(node2);
|
||||||
|
poolWithoutSize.TryPop(out _);
|
||||||
|
poolWithoutSize.TryPop(out _);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Benchmark]
|
||||||
|
public void TaskPoolWithoutLock()
|
||||||
|
{
|
||||||
|
poolWithoutLock.TryPush(node1);
|
||||||
|
poolWithoutLock.TryPush(node2);
|
||||||
|
poolWithoutLock.TryPop(out _);
|
||||||
|
poolWithoutLock.TryPop(out _);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public sealed class Node : ITaskPoolNode<Node>
|
||||||
|
{
|
||||||
|
public Node NextNode { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public interface ITaskPoolNode<T>
|
||||||
|
{
|
||||||
|
T NextNode { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
// mutable struct, don't mark readonly.
|
||||||
|
[StructLayout(LayoutKind.Auto)]
|
||||||
|
public struct TaskPoolWithoutLock<T>
|
||||||
|
where T : class, ITaskPoolNode<T>
|
||||||
|
{
|
||||||
|
int size;
|
||||||
|
T root;
|
||||||
|
|
||||||
|
public int Size => size;
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool TryPop(out T result)
|
||||||
|
{
|
||||||
|
//if (Interlocked.CompareExchange(ref gate, 1, 0) == 0)
|
||||||
|
{
|
||||||
|
var v = root;
|
||||||
|
if (!(v is null))
|
||||||
|
{
|
||||||
|
root = v.NextNode;
|
||||||
|
v.NextNode = null;
|
||||||
|
size--;
|
||||||
|
result = v;
|
||||||
|
// Volatile.Write(ref gate, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
//Volatile.Write(ref gate, 0);
|
||||||
|
}
|
||||||
|
result = default;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool TryPush(T item)
|
||||||
|
{
|
||||||
|
//if (Interlocked.CompareExchange(ref gate, 1, 0) == 0)
|
||||||
|
{
|
||||||
|
//if (size < TaskPool.MaxPoolSize)
|
||||||
|
{
|
||||||
|
item.NextNode = root;
|
||||||
|
root = item;
|
||||||
|
size++;
|
||||||
|
// Volatile.Write(ref gate, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
//else
|
||||||
|
{
|
||||||
|
// Volatile.Write(ref gate, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[StructLayout(LayoutKind.Auto)]
|
||||||
|
public struct TaskPool<T>
|
||||||
|
where T : class, ITaskPoolNode<T>
|
||||||
|
{
|
||||||
|
int gate;
|
||||||
|
int size;
|
||||||
|
T root;
|
||||||
|
|
||||||
|
public int Size => size;
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool TryPop(out T result)
|
||||||
|
{
|
||||||
|
if (Interlocked.CompareExchange(ref gate, 1, 0) == 0)
|
||||||
|
{
|
||||||
|
var v = root;
|
||||||
|
if (!(v is null))
|
||||||
|
{
|
||||||
|
root = v.NextNode;
|
||||||
|
v.NextNode = null;
|
||||||
|
size--;
|
||||||
|
result = v;
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
}
|
||||||
|
result = default;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool TryPush(T item)
|
||||||
|
{
|
||||||
|
if (Interlocked.CompareExchange(ref gate, 1, 0) == 0)
|
||||||
|
{
|
||||||
|
//if (size < TaskPool.MaxPoolSize)
|
||||||
|
{
|
||||||
|
item.NextNode = root;
|
||||||
|
root = item;
|
||||||
|
size++;
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
//else
|
||||||
|
{
|
||||||
|
// Volatile.Write(ref gate, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[StructLayout(LayoutKind.Auto)]
|
||||||
|
public struct TaskPoolEqualNull<T>
|
||||||
|
where T : class, ITaskPoolNode<T>
|
||||||
|
{
|
||||||
|
int gate;
|
||||||
|
int size;
|
||||||
|
T root;
|
||||||
|
|
||||||
|
public int Size => size;
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool TryPop(out T result)
|
||||||
|
{
|
||||||
|
if (Interlocked.CompareExchange(ref gate, 1, 0) == 0)
|
||||||
|
{
|
||||||
|
var v = root;
|
||||||
|
if (v != null)
|
||||||
|
{
|
||||||
|
root = v.NextNode;
|
||||||
|
v.NextNode = null;
|
||||||
|
size--;
|
||||||
|
result = v;
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
}
|
||||||
|
result = default;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool TryPush(T item)
|
||||||
|
{
|
||||||
|
if (Interlocked.CompareExchange(ref gate, 1, 0) == 0)
|
||||||
|
{
|
||||||
|
//if (size < TaskPool.MaxPoolSize)
|
||||||
|
{
|
||||||
|
item.NextNode = root;
|
||||||
|
root = item;
|
||||||
|
size++;
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
//else
|
||||||
|
{
|
||||||
|
// Volatile.Write(ref gate, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class TaskPoolClass<T>
|
||||||
|
where T : class, ITaskPoolNode<T>
|
||||||
|
{
|
||||||
|
int gate;
|
||||||
|
int size;
|
||||||
|
T root;
|
||||||
|
|
||||||
|
public int Size => size;
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool TryPop(out T result)
|
||||||
|
{
|
||||||
|
if (Interlocked.CompareExchange(ref gate, 1, 0) == 0)
|
||||||
|
{
|
||||||
|
var v = root;
|
||||||
|
if (!(v is null))
|
||||||
|
{
|
||||||
|
root = v.NextNode;
|
||||||
|
v.NextNode = null;
|
||||||
|
size--;
|
||||||
|
result = v;
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
}
|
||||||
|
result = default;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool TryPush(T item)
|
||||||
|
{
|
||||||
|
if (Interlocked.CompareExchange(ref gate, 1, 0) == 0)
|
||||||
|
{
|
||||||
|
//if (size < TaskPool.MaxPoolSize)
|
||||||
|
{
|
||||||
|
item.NextNode = root;
|
||||||
|
root = item;
|
||||||
|
size++;
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
//else
|
||||||
|
{
|
||||||
|
// Volatile.Write(ref gate, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[StructLayout(LayoutKind.Auto)]
|
||||||
|
public struct TaskPoolWithoutSize<T>
|
||||||
|
where T : class, ITaskPoolNode<T>
|
||||||
|
{
|
||||||
|
int gate;
|
||||||
|
T root;
|
||||||
|
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool TryPop(out T result)
|
||||||
|
{
|
||||||
|
if (Interlocked.CompareExchange(ref gate, 1, 0) == 0)
|
||||||
|
{
|
||||||
|
var v = root;
|
||||||
|
if (!(v is null))
|
||||||
|
{
|
||||||
|
root = v.NextNode;
|
||||||
|
v.NextNode = null;
|
||||||
|
result = v;
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
}
|
||||||
|
result = default;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool TryPush(T item)
|
||||||
|
{
|
||||||
|
if (Interlocked.CompareExchange(ref gate, 1, 0) == 0)
|
||||||
|
{
|
||||||
|
//if (size < TaskPool.MaxPoolSize)
|
||||||
|
{
|
||||||
|
item.NextNode = root;
|
||||||
|
root = item;
|
||||||
|
Volatile.Write(ref gate, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
//else
|
||||||
|
{
|
||||||
|
// Volatile.Write(ref gate, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -7,6 +7,8 @@
|
|||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
<PackageReference Include="BenchmarkDotNet" Version="0.12.1" />
|
||||||
|
<PackageReference Include="PooledAwait" Version="1.0.49" />
|
||||||
<PackageReference Include="System.Interactive.Async" Version="4.1.1" />
|
<PackageReference Include="System.Interactive.Async" Version="4.1.1" />
|
||||||
<PackageReference Include="System.Reactive" Version="4.4.1" />
|
<PackageReference Include="System.Reactive" Version="4.4.1" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|||||||
118
src/UniTask.NetCoreTests/AsyncReactivePropertyTest.cs
Normal file
118
src/UniTask.NetCoreTests/AsyncReactivePropertyTest.cs
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using FluentAssertions;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Channels;
|
||||||
|
using Cysharp.Threading.Tasks.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace NetCoreTests
|
||||||
|
{
|
||||||
|
public class AsyncReactivePropertyTest
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public async Task Iteration()
|
||||||
|
{
|
||||||
|
var rp = new AsyncReactiveProperty<int>(99);
|
||||||
|
|
||||||
|
var f = await rp.FirstAsync();
|
||||||
|
f.Should().Be(99);
|
||||||
|
|
||||||
|
var array = rp.Take(5).ToArrayAsync();
|
||||||
|
|
||||||
|
rp.Value = 100;
|
||||||
|
rp.Value = 100;
|
||||||
|
rp.Value = 100;
|
||||||
|
rp.Value = 131;
|
||||||
|
|
||||||
|
var ar = await array;
|
||||||
|
|
||||||
|
ar.Should().BeEquivalentTo(new[] { 99, 100, 100, 100, 131 });
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task WithoutCurrent()
|
||||||
|
{
|
||||||
|
var rp = new AsyncReactiveProperty<int>(99);
|
||||||
|
|
||||||
|
var array = rp.WithoutCurrent().Take(5).ToArrayAsync();
|
||||||
|
|
||||||
|
rp.Value = 100;
|
||||||
|
rp.Value = 100;
|
||||||
|
rp.Value = 100;
|
||||||
|
rp.Value = 131;
|
||||||
|
rp.Value = 191;
|
||||||
|
|
||||||
|
var ar = await array;
|
||||||
|
|
||||||
|
ar.Should().BeEquivalentTo(new[] { 100, 100, 100, 131, 191 });
|
||||||
|
}
|
||||||
|
|
||||||
|
//[Fact]
|
||||||
|
//public async Task StateIteration()
|
||||||
|
//{
|
||||||
|
// var rp = new ReadOnlyAsyncReactiveProperty<int>(99);
|
||||||
|
// var setter = rp.GetSetter();
|
||||||
|
|
||||||
|
// var f = await rp.FirstAsync();
|
||||||
|
// f.Should().Be(99);
|
||||||
|
|
||||||
|
// var array = rp.Take(5).ToArrayAsync();
|
||||||
|
|
||||||
|
// setter(100);
|
||||||
|
// setter(100);
|
||||||
|
// setter(100);
|
||||||
|
// setter(131);
|
||||||
|
|
||||||
|
// var ar = await array;
|
||||||
|
|
||||||
|
// ar.Should().BeEquivalentTo(new[] { 99, 100, 100, 100, 131 });
|
||||||
|
//}
|
||||||
|
|
||||||
|
//[Fact]
|
||||||
|
//public async Task StateWithoutCurrent()
|
||||||
|
//{
|
||||||
|
// var rp = new ReadOnlyAsyncReactiveProperty<int>(99);
|
||||||
|
// var setter = rp.GetSetter();
|
||||||
|
|
||||||
|
// var array = rp.WithoutCurrent().Take(5).ToArrayAsync();
|
||||||
|
// setter(100);
|
||||||
|
// setter(100);
|
||||||
|
// setter(100);
|
||||||
|
// setter(131);
|
||||||
|
// setter(191);
|
||||||
|
|
||||||
|
// var ar = await array;
|
||||||
|
|
||||||
|
// ar.Should().BeEquivalentTo(new[] { 100, 100, 100, 131, 191 });
|
||||||
|
//}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void StateFromEnumeration()
|
||||||
|
{
|
||||||
|
var rp = new AsyncReactiveProperty<int>(10);
|
||||||
|
|
||||||
|
var state = rp.ToReadOnlyAsyncReactiveProperty(CancellationToken.None);
|
||||||
|
|
||||||
|
rp.Value = 10;
|
||||||
|
state.Value.Should().Be(10);
|
||||||
|
|
||||||
|
rp.Value = 20;
|
||||||
|
state.Value.Should().Be(20);
|
||||||
|
|
||||||
|
state.Dispose();
|
||||||
|
|
||||||
|
rp.Value = 30;
|
||||||
|
state.Value.Should().Be(20);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
51
src/UniTask.NetCoreTests/CancellationTokenTest.cs
Normal file
51
src/UniTask.NetCoreTests/CancellationTokenTest.cs
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using FluentAssertions;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Channels;
|
||||||
|
using Cysharp.Threading.Tasks.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace NetCoreTests
|
||||||
|
{
|
||||||
|
public class CancellationTokenTest
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public async Task WaitUntilCanceled()
|
||||||
|
{
|
||||||
|
var cts = new CancellationTokenSource();
|
||||||
|
|
||||||
|
cts.CancelAfter(TimeSpan.FromSeconds(1.5));
|
||||||
|
|
||||||
|
var now = DateTime.UtcNow;
|
||||||
|
|
||||||
|
await cts.Token.WaitUntilCanceled();
|
||||||
|
|
||||||
|
var elapsed = DateTime.UtcNow - now;
|
||||||
|
|
||||||
|
elapsed.Should().BeGreaterThan(TimeSpan.FromSeconds(1));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void AlreadyCanceled()
|
||||||
|
{
|
||||||
|
var cts = new CancellationTokenSource();
|
||||||
|
|
||||||
|
cts.Cancel();
|
||||||
|
|
||||||
|
cts.Token.WaitUntilCanceled().GetAwaiter().IsCompleted.Should().BeTrue();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void None()
|
||||||
|
{
|
||||||
|
CancellationToken.None.WaitUntilCanceled().GetAwaiter().IsCompleted.Should().BeTrue();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
370
src/UniTask.NetCoreTests/ChannelTest.cs
Normal file
370
src/UniTask.NetCoreTests/ChannelTest.cs
Normal file
@@ -0,0 +1,370 @@
|
|||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using FluentAssertions;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Channels;
|
||||||
|
using Cysharp.Threading.Tasks.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace NetCoreTests
|
||||||
|
{
|
||||||
|
public class ChannelTest
|
||||||
|
{
|
||||||
|
(System.Threading.Channels.Channel<int>, Cysharp.Threading.Tasks.Channel<int>) CreateChannel()
|
||||||
|
{
|
||||||
|
var reference = System.Threading.Channels.Channel.CreateUnbounded<int>(new UnboundedChannelOptions
|
||||||
|
{
|
||||||
|
AllowSynchronousContinuations = true,
|
||||||
|
SingleReader = true,
|
||||||
|
SingleWriter = false
|
||||||
|
});
|
||||||
|
|
||||||
|
var channel = Cysharp.Threading.Tasks.Channel.CreateSingleConsumerUnbounded<int>();
|
||||||
|
|
||||||
|
return (reference, channel);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task SingleWriteSingleRead()
|
||||||
|
{
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
var t1 = reference.Reader.WaitToReadAsync();
|
||||||
|
var t2 = channel.Reader.WaitToReadAsync();
|
||||||
|
|
||||||
|
t1.IsCompleted.Should().BeFalse();
|
||||||
|
t2.Status.IsCompleted().Should().BeFalse();
|
||||||
|
|
||||||
|
reference.Writer.TryWrite(item);
|
||||||
|
channel.Writer.TryWrite(item);
|
||||||
|
|
||||||
|
(await t1).Should().BeTrue();
|
||||||
|
(await t2).Should().BeTrue();
|
||||||
|
|
||||||
|
reference.Reader.TryRead(out var refitem).Should().BeTrue();
|
||||||
|
channel.Reader.TryRead(out var chanitem).Should().BeTrue();
|
||||||
|
refitem.Should().Be(item);
|
||||||
|
chanitem.Should().Be(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task MultiWrite()
|
||||||
|
{
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
var t1 = reference.Reader.WaitToReadAsync();
|
||||||
|
var t2 = channel.Reader.WaitToReadAsync();
|
||||||
|
|
||||||
|
t1.IsCompleted.Should().BeFalse();
|
||||||
|
t2.Status.IsCompleted().Should().BeFalse();
|
||||||
|
|
||||||
|
foreach (var i in Enumerable.Range(1, 3))
|
||||||
|
{
|
||||||
|
reference.Writer.TryWrite(item * i);
|
||||||
|
channel.Writer.TryWrite(item * i);
|
||||||
|
}
|
||||||
|
|
||||||
|
(await t1).Should().BeTrue();
|
||||||
|
(await t2).Should().BeTrue();
|
||||||
|
|
||||||
|
foreach (var i in Enumerable.Range(1, 3))
|
||||||
|
{
|
||||||
|
(await reference.Reader.WaitToReadAsync()).Should().BeTrue();
|
||||||
|
(await channel.Reader.WaitToReadAsync()).Should().BeTrue();
|
||||||
|
|
||||||
|
reference.Reader.TryRead(out var refitem).Should().BeTrue();
|
||||||
|
channel.Reader.TryRead(out var chanitem).Should().BeTrue();
|
||||||
|
refitem.Should().Be(item * i);
|
||||||
|
chanitem.Should().Be(item * i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task CompleteOnEmpty()
|
||||||
|
{
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
reference.Writer.TryWrite(item);
|
||||||
|
channel.Writer.TryWrite(item);
|
||||||
|
reference.Reader.TryRead(out var refitem);
|
||||||
|
channel.Reader.TryRead(out var chanitem);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Empty.
|
||||||
|
|
||||||
|
var completion1 = reference.Reader.Completion;
|
||||||
|
var wait1 = reference.Reader.WaitToReadAsync();
|
||||||
|
|
||||||
|
var completion2 = channel.Reader.Completion;
|
||||||
|
var wait2 = channel.Reader.WaitToReadAsync();
|
||||||
|
|
||||||
|
reference.Writer.TryComplete();
|
||||||
|
channel.Writer.TryComplete();
|
||||||
|
|
||||||
|
completion1.Status.Should().Be(TaskStatus.RanToCompletion);
|
||||||
|
completion2.Status.Should().Be(UniTaskStatus.Succeeded);
|
||||||
|
|
||||||
|
(await wait1).Should().BeFalse();
|
||||||
|
(await wait2).Should().BeFalse();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task CompleteErrorOnEmpty()
|
||||||
|
{
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
reference.Writer.TryWrite(item);
|
||||||
|
channel.Writer.TryWrite(item);
|
||||||
|
reference.Reader.TryRead(out var refitem);
|
||||||
|
channel.Reader.TryRead(out var chanitem);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Empty.
|
||||||
|
|
||||||
|
var completion1 = reference.Reader.Completion;
|
||||||
|
var wait1 = reference.Reader.WaitToReadAsync();
|
||||||
|
|
||||||
|
var completion2 = channel.Reader.Completion;
|
||||||
|
var wait2 = channel.Reader.WaitToReadAsync();
|
||||||
|
|
||||||
|
var ex = new Exception();
|
||||||
|
reference.Writer.TryComplete(ex);
|
||||||
|
channel.Writer.TryComplete(ex);
|
||||||
|
|
||||||
|
completion1.Status.Should().Be(TaskStatus.Faulted);
|
||||||
|
completion2.Status.Should().Be(UniTaskStatus.Faulted);
|
||||||
|
|
||||||
|
(await Assert.ThrowsAsync<Exception>(async () => await wait1)).Should().Be(ex);
|
||||||
|
(await Assert.ThrowsAsync<Exception>(async () => await wait2)).Should().Be(ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task CompleteWithRest()
|
||||||
|
{
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
reference.Writer.TryWrite(item);
|
||||||
|
channel.Writer.TryWrite(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Three Item2.
|
||||||
|
|
||||||
|
var completion1 = reference.Reader.Completion;
|
||||||
|
var wait1 = reference.Reader.WaitToReadAsync();
|
||||||
|
|
||||||
|
var completion2 = channel.Reader.Completion;
|
||||||
|
var wait2 = channel.Reader.WaitToReadAsync();
|
||||||
|
|
||||||
|
reference.Writer.TryComplete();
|
||||||
|
channel.Writer.TryComplete();
|
||||||
|
|
||||||
|
// completion1.Status.Should().Be(TaskStatus.WaitingForActivation);
|
||||||
|
completion2.Status.Should().Be(UniTaskStatus.Pending);
|
||||||
|
|
||||||
|
(await wait1).Should().BeTrue();
|
||||||
|
(await wait2).Should().BeTrue();
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
reference.Reader.TryRead(out var i1).Should().BeTrue();
|
||||||
|
channel.Reader.TryRead(out var i2).Should().BeTrue();
|
||||||
|
i1.Should().Be(item);
|
||||||
|
i2.Should().Be(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
(await reference.Reader.WaitToReadAsync()).Should().BeFalse();
|
||||||
|
(await channel.Reader.WaitToReadAsync()).Should().BeFalse();
|
||||||
|
|
||||||
|
completion1.Status.Should().Be(TaskStatus.RanToCompletion);
|
||||||
|
completion2.Status.Should().Be(UniTaskStatus.Succeeded);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task CompleteErrorWithRest()
|
||||||
|
{
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
reference.Writer.TryWrite(item);
|
||||||
|
channel.Writer.TryWrite(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Three Item2.
|
||||||
|
|
||||||
|
var completion1 = reference.Reader.Completion;
|
||||||
|
var wait1 = reference.Reader.WaitToReadAsync();
|
||||||
|
|
||||||
|
var completion2 = channel.Reader.Completion;
|
||||||
|
var wait2 = channel.Reader.WaitToReadAsync();
|
||||||
|
|
||||||
|
var ex = new Exception();
|
||||||
|
reference.Writer.TryComplete(ex);
|
||||||
|
channel.Writer.TryComplete(ex);
|
||||||
|
|
||||||
|
// completion1.Status.Should().Be(TaskStatus.WaitingForActivation);
|
||||||
|
completion2.Status.Should().Be(UniTaskStatus.Pending);
|
||||||
|
|
||||||
|
(await wait1).Should().BeTrue();
|
||||||
|
(await wait2).Should().BeTrue();
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
reference.Reader.TryRead(out var i1).Should().BeTrue();
|
||||||
|
channel.Reader.TryRead(out var i2).Should().BeTrue();
|
||||||
|
i1.Should().Be(item);
|
||||||
|
i2.Should().Be(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
wait1 = reference.Reader.WaitToReadAsync();
|
||||||
|
wait2 = channel.Reader.WaitToReadAsync();
|
||||||
|
|
||||||
|
(await Assert.ThrowsAsync<Exception>(async () => await wait1)).Should().Be(ex);
|
||||||
|
(await Assert.ThrowsAsync<Exception>(async () => await wait2)).Should().Be(ex);
|
||||||
|
|
||||||
|
completion1.Status.Should().Be(TaskStatus.Faulted);
|
||||||
|
completion2.Status.Should().Be(UniTaskStatus.Faulted);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task Cancellation()
|
||||||
|
{
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
var cts = new CancellationTokenSource();
|
||||||
|
|
||||||
|
var wait1 = reference.Reader.WaitToReadAsync(cts.Token);
|
||||||
|
var wait2 = channel.Reader.WaitToReadAsync(cts.Token);
|
||||||
|
|
||||||
|
cts.Cancel();
|
||||||
|
|
||||||
|
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await wait1)).CancellationToken.Should().Be(cts.Token);
|
||||||
|
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await wait2)).CancellationToken.Should().Be(cts.Token);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task AsyncEnumerator()
|
||||||
|
{
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
var ta1 = reference.Reader.ReadAllAsync().ToArrayAsync();
|
||||||
|
var ta2 = channel.Reader.ReadAllAsync().ToArrayAsync();
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
reference.Writer.TryWrite(item);
|
||||||
|
channel.Writer.TryWrite(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
reference.Writer.TryComplete();
|
||||||
|
channel.Writer.TryComplete();
|
||||||
|
|
||||||
|
(await ta1).Should().BeEquivalentTo(new[] { 10, 20, 30 });
|
||||||
|
(await ta2).Should().BeEquivalentTo(new[] { 10, 20, 30 });
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task AsyncEnumeratorCancellation()
|
||||||
|
{
|
||||||
|
// Token1, Token2 and Cancel1
|
||||||
|
{
|
||||||
|
var cts1 = new CancellationTokenSource();
|
||||||
|
var cts2 = new CancellationTokenSource();
|
||||||
|
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
var ta1 = reference.Reader.ReadAllAsync(cts1.Token).ToArrayAsync(cts2.Token);
|
||||||
|
var ta2 = channel.Reader.ReadAllAsync(cts1.Token).ToArrayAsync(cts2.Token);
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
reference.Writer.TryWrite(item);
|
||||||
|
channel.Writer.TryWrite(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
cts1.Cancel();
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta1);
|
||||||
|
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta2)).CancellationToken.Should().Be(cts1.Token);
|
||||||
|
}
|
||||||
|
// Token1, Token2 and Cancel2
|
||||||
|
{
|
||||||
|
var cts1 = new CancellationTokenSource();
|
||||||
|
var cts2 = new CancellationTokenSource();
|
||||||
|
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
var ta1 = reference.Reader.ReadAllAsync(cts1.Token).ToArrayAsync(cts2.Token);
|
||||||
|
var ta2 = channel.Reader.ReadAllAsync(cts1.Token).ToArrayAsync(cts2.Token);
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
reference.Writer.TryWrite(item);
|
||||||
|
channel.Writer.TryWrite(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
cts2.Cancel();
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta1);
|
||||||
|
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta2)).CancellationToken.Should().Be(cts2.Token);
|
||||||
|
}
|
||||||
|
// Token1 and Cancel1
|
||||||
|
{
|
||||||
|
var cts1 = new CancellationTokenSource();
|
||||||
|
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
var ta1 = reference.Reader.ReadAllAsync(cts1.Token).ToArrayAsync();
|
||||||
|
var ta2 = channel.Reader.ReadAllAsync(cts1.Token).ToArrayAsync();
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
reference.Writer.TryWrite(item);
|
||||||
|
channel.Writer.TryWrite(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
cts1.Cancel();
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta1);
|
||||||
|
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta2)).CancellationToken.Should().Be(cts1.Token);
|
||||||
|
}
|
||||||
|
// Token2 and Cancel2
|
||||||
|
{
|
||||||
|
var cts2 = new CancellationTokenSource();
|
||||||
|
|
||||||
|
var (reference, channel) = CreateChannel();
|
||||||
|
|
||||||
|
var ta1 = reference.Reader.ReadAllAsync().ToArrayAsync(cts2.Token);
|
||||||
|
var ta2 = channel.Reader.ReadAllAsync().ToArrayAsync(cts2.Token);
|
||||||
|
|
||||||
|
foreach (var item in new[] { 10, 20, 30 })
|
||||||
|
{
|
||||||
|
reference.Writer.TryWrite(item);
|
||||||
|
channel.Writer.TryWrite(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
cts2.Cancel();
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta1);
|
||||||
|
(await Assert.ThrowsAsync<OperationCanceledException>(async () => await ta2)).CancellationToken.Should().Be(cts2.Token);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
47
src/UniTask.NetCoreTests/DeferTest.cs
Normal file
47
src/UniTask.NetCoreTests/DeferTest.cs
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using FluentAssertions;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Channels;
|
||||||
|
using Cysharp.Threading.Tasks.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace NetCoreTests
|
||||||
|
{
|
||||||
|
public class DeferTest
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public async Task D()
|
||||||
|
{
|
||||||
|
var created = false;
|
||||||
|
var v = UniTask.Defer(() => { created = true; return UniTask.Run(() => 10); });
|
||||||
|
|
||||||
|
created.Should().BeFalse();
|
||||||
|
|
||||||
|
var t = await v;
|
||||||
|
|
||||||
|
created.Should().BeTrue();
|
||||||
|
|
||||||
|
t.Should().Be(10);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task D2()
|
||||||
|
{
|
||||||
|
var created = false;
|
||||||
|
var v = UniTask.Defer(() => { created = true; return UniTask.Run(() => 10).AsUniTask(); });
|
||||||
|
|
||||||
|
created.Should().BeFalse();
|
||||||
|
|
||||||
|
await v;
|
||||||
|
|
||||||
|
created.Should().BeTrue();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
@@ -35,7 +35,7 @@ namespace NetCoreTests.Linq
|
|||||||
xs.Should().Be(ys);
|
xs.Should().Be(ys);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
var xs = await UniTaskAsyncEnumerable.Range(start, count).SumAwaitCancellationAsync((x, _) => UniTask.Run(() => x));
|
var xs = await UniTaskAsyncEnumerable.Range(start, count).SumAwaitWithCancellationAsync((x, _) => UniTask.Run(() => x));
|
||||||
var ys = Enumerable.Range(start, count).Sum(x => x);
|
var ys = Enumerable.Range(start, count).Sum(x => x);
|
||||||
xs.Should().Be(ys);
|
xs.Should().Be(ys);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -319,5 +319,119 @@ namespace NetCoreTests.Linq
|
|||||||
await Assert.ThrowsAsync<UniTaskTestException>(async () => await ys);
|
await Assert.ThrowsAsync<UniTaskTestException>(async () => await ys);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task CombineLatestOK()
|
||||||
|
{
|
||||||
|
var a = new AsyncReactiveProperty<int>(0);
|
||||||
|
var b = new AsyncReactiveProperty<int>(0);
|
||||||
|
|
||||||
|
var list = new List<(int, int)>();
|
||||||
|
var complete = a.WithoutCurrent().CombineLatest(b.WithoutCurrent(), (x, y) => (x, y)).ForEachAsync(x => list.Add(x));
|
||||||
|
|
||||||
|
list.Count.Should().Be(0);
|
||||||
|
|
||||||
|
a.Value = 10;
|
||||||
|
list.Count.Should().Be(0);
|
||||||
|
|
||||||
|
a.Value = 20;
|
||||||
|
list.Count.Should().Be(0);
|
||||||
|
|
||||||
|
b.Value = 1;
|
||||||
|
list.Count.Should().Be(1);
|
||||||
|
|
||||||
|
list[0].Should().Be((20, 1));
|
||||||
|
|
||||||
|
a.Value = 30;
|
||||||
|
list.Last().Should().Be((30, 1));
|
||||||
|
|
||||||
|
b.Value = 2;
|
||||||
|
list.Last().Should().Be((30, 2));
|
||||||
|
|
||||||
|
a.Dispose();
|
||||||
|
b.Value = 3;
|
||||||
|
list.Last().Should().Be((30, 3));
|
||||||
|
|
||||||
|
b.Dispose();
|
||||||
|
|
||||||
|
await complete;
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task CombineLatestLong()
|
||||||
|
{
|
||||||
|
var a = UniTaskAsyncEnumerable.Range(1, 100000);
|
||||||
|
var b = new AsyncReactiveProperty<int>(0);
|
||||||
|
|
||||||
|
var list = new List<(int, int)>();
|
||||||
|
var complete = a.CombineLatest(b.WithoutCurrent(), (x, y) => (x, y)).ForEachAsync(x => list.Add(x));
|
||||||
|
|
||||||
|
b.Value = 1;
|
||||||
|
|
||||||
|
list[0].Should().Be((100000, 1));
|
||||||
|
|
||||||
|
b.Dispose();
|
||||||
|
|
||||||
|
await complete;
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task CombineLatestError()
|
||||||
|
{
|
||||||
|
var a = new AsyncReactiveProperty<int>(0);
|
||||||
|
var b = new AsyncReactiveProperty<int>(0);
|
||||||
|
|
||||||
|
var list = new List<(int, int)>();
|
||||||
|
var complete = a.WithoutCurrent()
|
||||||
|
.Select(x => { if (x == 0) { throw new MyException(); } return x; })
|
||||||
|
.CombineLatest(b.WithoutCurrent(), (x, y) => (x, y)).ForEachAsync(x => list.Add(x));
|
||||||
|
|
||||||
|
|
||||||
|
a.Value = 10;
|
||||||
|
b.Value = 1;
|
||||||
|
list.Last().Should().Be((10, 1));
|
||||||
|
|
||||||
|
a.Value = 0;
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<MyException>(async () => await complete);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task PariwiseImmediate()
|
||||||
|
{
|
||||||
|
var xs = await UniTaskAsyncEnumerable.Range(1, 5).Pairwise().ToArrayAsync();
|
||||||
|
xs.Should().BeEquivalentTo((1, 2), (2, 3), (3, 4), (4, 5));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task Pariwise()
|
||||||
|
{
|
||||||
|
var a = new AsyncReactiveProperty<int>(0);
|
||||||
|
|
||||||
|
var list = new List<(int, int)>();
|
||||||
|
var complete = a.WithoutCurrent().Pairwise().ForEachAsync(x => list.Add(x));
|
||||||
|
|
||||||
|
list.Count.Should().Be(0);
|
||||||
|
a.Value = 10;
|
||||||
|
list.Count.Should().Be(0);
|
||||||
|
a.Value = 20;
|
||||||
|
list.Count.Should().Be(1);
|
||||||
|
a.Value = 30;
|
||||||
|
a.Value = 40;
|
||||||
|
a.Value = 50;
|
||||||
|
|
||||||
|
a.Dispose();
|
||||||
|
|
||||||
|
await complete;
|
||||||
|
|
||||||
|
list.Should().BeEquivalentTo((10, 20), (20, 30), (30, 40), (40, 50));
|
||||||
|
}
|
||||||
|
|
||||||
|
class MyException : Exception
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
78
src/UniTask.NetCoreTests/Linq/PulbishTest.cs
Normal file
78
src/UniTask.NetCoreTests/Linq/PulbishTest.cs
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using Cysharp.Threading.Tasks.Linq;
|
||||||
|
using FluentAssertions;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace NetCoreTests.Linq
|
||||||
|
{
|
||||||
|
public class PublishTest
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public async Task Normal()
|
||||||
|
{
|
||||||
|
var rp = new AsyncReactiveProperty<int>(1);
|
||||||
|
|
||||||
|
var multicast = rp.Publish();
|
||||||
|
|
||||||
|
var a = multicast.ToArrayAsync();
|
||||||
|
var b = multicast.Take(2).ToArrayAsync();
|
||||||
|
|
||||||
|
var disp = multicast.Connect();
|
||||||
|
|
||||||
|
rp.Value = 2;
|
||||||
|
|
||||||
|
(await b).Should().BeEquivalentTo(1, 2);
|
||||||
|
|
||||||
|
var c = multicast.ToArrayAsync();
|
||||||
|
|
||||||
|
rp.Value = 3;
|
||||||
|
rp.Value = 4;
|
||||||
|
rp.Value = 5;
|
||||||
|
|
||||||
|
rp.Dispose();
|
||||||
|
|
||||||
|
(await a).Should().BeEquivalentTo(1, 2, 3, 4, 5);
|
||||||
|
(await c).Should().BeEquivalentTo(3, 4, 5);
|
||||||
|
|
||||||
|
disp.Dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task Cancel()
|
||||||
|
{
|
||||||
|
var rp = new AsyncReactiveProperty<int>(1);
|
||||||
|
|
||||||
|
var multicast = rp.Publish();
|
||||||
|
|
||||||
|
var a = multicast.ToArrayAsync();
|
||||||
|
var b = multicast.Take(2).ToArrayAsync();
|
||||||
|
|
||||||
|
var disp = multicast.Connect();
|
||||||
|
|
||||||
|
rp.Value = 2;
|
||||||
|
|
||||||
|
(await b).Should().BeEquivalentTo(1, 2);
|
||||||
|
|
||||||
|
var c = multicast.ToArrayAsync();
|
||||||
|
|
||||||
|
rp.Value = 3;
|
||||||
|
|
||||||
|
disp.Dispose();
|
||||||
|
|
||||||
|
rp.Value = 4;
|
||||||
|
rp.Value = 5;
|
||||||
|
|
||||||
|
rp.Dispose();
|
||||||
|
|
||||||
|
await Assert.ThrowsAsync<OperationCanceledException>(async () => await a);
|
||||||
|
await Assert.ThrowsAsync<OperationCanceledException>(async () => await c);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
29
src/UniTask.NetCoreTests/Linq/QueueTest.cs
Normal file
29
src/UniTask.NetCoreTests/Linq/QueueTest.cs
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using Cysharp.Threading.Tasks.Linq;
|
||||||
|
using FluentAssertions;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace NetCoreTests.Linq
|
||||||
|
{
|
||||||
|
public class QueueTest
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public async Task Q()
|
||||||
|
{
|
||||||
|
var rp = new AsyncReactiveProperty<int>(100);
|
||||||
|
|
||||||
|
var l = new List<int>();
|
||||||
|
await rp.Take(10).Queue().ForEachAsync(x =>
|
||||||
|
{
|
||||||
|
rp.Value += 10;
|
||||||
|
l.Add(x);
|
||||||
|
});
|
||||||
|
|
||||||
|
l.Should().BeEquivalentTo(100, 110, 120, 130, 140, 150, 160, 170, 180, 190);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
106
src/UniTask.NetCoreTests/Linq/TakeInfinityTest.cs
Normal file
106
src/UniTask.NetCoreTests/Linq/TakeInfinityTest.cs
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using Cysharp.Threading.Tasks.Linq;
|
||||||
|
using FluentAssertions;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace NetCoreTests.Linq
|
||||||
|
{
|
||||||
|
public class TakeInfinityTest
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public async Task Take()
|
||||||
|
{
|
||||||
|
var rp = new AsyncReactiveProperty<int>(1);
|
||||||
|
|
||||||
|
var xs = rp.Take(5).ToArrayAsync();
|
||||||
|
|
||||||
|
rp.Value = 2;
|
||||||
|
rp.Value = 3;
|
||||||
|
rp.Value = 4;
|
||||||
|
rp.Value = 5;
|
||||||
|
|
||||||
|
(await xs).Should().BeEquivalentTo(1, 2, 3, 4, 5);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task TakeWhile()
|
||||||
|
{
|
||||||
|
var rp = new AsyncReactiveProperty<int>(1);
|
||||||
|
|
||||||
|
var xs = rp.TakeWhile(x => x != 5).ToArrayAsync();
|
||||||
|
|
||||||
|
rp.Value = 2;
|
||||||
|
rp.Value = 3;
|
||||||
|
rp.Value = 4;
|
||||||
|
rp.Value = 5;
|
||||||
|
|
||||||
|
(await xs).Should().BeEquivalentTo(1, 2, 3, 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task TakeUntil()
|
||||||
|
{
|
||||||
|
var cts = new CancellationTokenSource();
|
||||||
|
|
||||||
|
var rp = new AsyncReactiveProperty<int>(1);
|
||||||
|
|
||||||
|
var xs = rp.TakeUntilCanceled(cts.Token).ToArrayAsync();
|
||||||
|
|
||||||
|
var c = CancelAsync();
|
||||||
|
|
||||||
|
await c;
|
||||||
|
var foo = await xs;
|
||||||
|
|
||||||
|
foo.Should().BeEquivalentTo(new[] { 1, 10, 20 });
|
||||||
|
|
||||||
|
async Task CancelAsync()
|
||||||
|
{
|
||||||
|
rp.Value = 10;
|
||||||
|
await Task.Yield();
|
||||||
|
rp.Value = 20;
|
||||||
|
await Task.Yield();
|
||||||
|
cts.Cancel();
|
||||||
|
rp.Value = 30;
|
||||||
|
await Task.Yield();
|
||||||
|
rp.Value = 40;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task SkipUntil()
|
||||||
|
{
|
||||||
|
var cts = new CancellationTokenSource();
|
||||||
|
|
||||||
|
var rp = new AsyncReactiveProperty<int>(1);
|
||||||
|
|
||||||
|
var xs = rp.SkipUntilCanceled(cts.Token).ToArrayAsync();
|
||||||
|
|
||||||
|
var c = CancelAsync();
|
||||||
|
|
||||||
|
await c;
|
||||||
|
var foo = await xs;
|
||||||
|
|
||||||
|
foo.Should().BeEquivalentTo(new[] { 30, 40 });
|
||||||
|
|
||||||
|
async Task CancelAsync()
|
||||||
|
{
|
||||||
|
rp.Value = 10;
|
||||||
|
await Task.Yield();
|
||||||
|
rp.Value = 20;
|
||||||
|
await Task.Yield();
|
||||||
|
cts.Cancel();
|
||||||
|
rp.Value = 30;
|
||||||
|
await Task.Yield();
|
||||||
|
rp.Value = 40;
|
||||||
|
|
||||||
|
rp.Dispose(); // complete.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
301
src/UniTask.NetCoreTests/TaskBuilderCases.cs
Normal file
301
src/UniTask.NetCoreTests/TaskBuilderCases.cs
Normal file
@@ -0,0 +1,301 @@
|
|||||||
|
#pragma warning disable CS1998
|
||||||
|
|
||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using FluentAssertions;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Channels;
|
||||||
|
using Cysharp.Threading.Tasks.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Xunit;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
|
||||||
|
namespace NetCoreTests
|
||||||
|
{
|
||||||
|
public class UniTaskBuilderTest
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public async Task Empty()
|
||||||
|
{
|
||||||
|
await Core();
|
||||||
|
|
||||||
|
static async UniTask Core()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task EmptyThrow()
|
||||||
|
{
|
||||||
|
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
||||||
|
|
||||||
|
static async UniTask Core()
|
||||||
|
{
|
||||||
|
throw new TaskTestException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task Task_Done()
|
||||||
|
{
|
||||||
|
await Core();
|
||||||
|
|
||||||
|
static async UniTask Core()
|
||||||
|
{
|
||||||
|
await new TestAwaiter(true, UniTaskStatus.Succeeded);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task Task_Fail()
|
||||||
|
{
|
||||||
|
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
||||||
|
|
||||||
|
static async UniTask Core()
|
||||||
|
{
|
||||||
|
await new TestAwaiter(true, UniTaskStatus.Faulted);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task Task_Cancel()
|
||||||
|
{
|
||||||
|
await Assert.ThrowsAsync<OperationCanceledException>(async () => await Core());
|
||||||
|
|
||||||
|
static async UniTask Core()
|
||||||
|
{
|
||||||
|
await new TestAwaiter(true, UniTaskStatus.Canceled);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task AwaitUnsafeOnCompletedCall_Task_SetResult()
|
||||||
|
{
|
||||||
|
await Core();
|
||||||
|
|
||||||
|
static async UniTask Core()
|
||||||
|
{
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task AwaitUnsafeOnCompletedCall_Task_SetException()
|
||||||
|
{
|
||||||
|
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
||||||
|
|
||||||
|
static async UniTask Core()
|
||||||
|
{
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Faulted);
|
||||||
|
throw new InvalidOperationException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task AwaitUnsafeOnCompletedCall_Task_SetCancelException()
|
||||||
|
{
|
||||||
|
await Assert.ThrowsAsync<OperationCanceledException>(async () => await Core());
|
||||||
|
|
||||||
|
static async UniTask Core()
|
||||||
|
{
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Succeeded);
|
||||||
|
await new TestAwaiter(false, UniTaskStatus.Canceled);
|
||||||
|
throw new InvalidOperationException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class UniTask_T_BuilderTest
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public async Task Empty()
|
||||||
|
{
|
||||||
|
(await Core()).Should().Be(10);
|
||||||
|
|
||||||
|
static async UniTask<int> Core()
|
||||||
|
{
|
||||||
|
return 10;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task EmptyThrow()
|
||||||
|
{
|
||||||
|
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
||||||
|
|
||||||
|
static async UniTask<int> Core()
|
||||||
|
{
|
||||||
|
throw new TaskTestException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task Task_Done()
|
||||||
|
{
|
||||||
|
(await Core()).Should().Be(10);
|
||||||
|
|
||||||
|
static async UniTask<int> Core()
|
||||||
|
{
|
||||||
|
return await new TestAwaiter<int>(true, UniTaskStatus.Succeeded, 10);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task Task_Fail()
|
||||||
|
{
|
||||||
|
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
||||||
|
|
||||||
|
static async UniTask<int> Core()
|
||||||
|
{
|
||||||
|
return await new TestAwaiter<int>(true, UniTaskStatus.Faulted, 10);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task Task_Cancel()
|
||||||
|
{
|
||||||
|
await Assert.ThrowsAsync<OperationCanceledException>(async () => await Core());
|
||||||
|
|
||||||
|
static async UniTask<int> Core()
|
||||||
|
{
|
||||||
|
return await new TestAwaiter<int>(true, UniTaskStatus.Canceled, 10);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task AwaitUnsafeOnCompletedCall_Task_SetResult()
|
||||||
|
{
|
||||||
|
(await Core()).Should().Be(6);
|
||||||
|
|
||||||
|
static async UniTask<int> Core()
|
||||||
|
{
|
||||||
|
var sum = 0;
|
||||||
|
sum += await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 1);
|
||||||
|
sum += await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 2);
|
||||||
|
sum += await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 3);
|
||||||
|
return sum;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task AwaitUnsafeOnCompletedCall_Task_SetException()
|
||||||
|
{
|
||||||
|
await Assert.ThrowsAsync<TaskTestException>(async () => await Core());
|
||||||
|
|
||||||
|
static async UniTask<int> Core()
|
||||||
|
{
|
||||||
|
await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 10);
|
||||||
|
await new TestAwaiter<int>(false, UniTaskStatus.Faulted, 10);
|
||||||
|
throw new InvalidOperationException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public async Task AwaitUnsafeOnCompletedCall_Task_SetCancelException()
|
||||||
|
{
|
||||||
|
await Assert.ThrowsAsync<OperationCanceledException>(async () => await Core());
|
||||||
|
|
||||||
|
static async UniTask<int> Core()
|
||||||
|
{
|
||||||
|
await new TestAwaiter<int>(false, UniTaskStatus.Succeeded, 10);
|
||||||
|
await new TestAwaiter<int>(false, UniTaskStatus.Canceled, 10);
|
||||||
|
throw new InvalidOperationException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class TaskTestException : Exception
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct TestAwaiter : ICriticalNotifyCompletion
|
||||||
|
{
|
||||||
|
readonly UniTaskStatus status;
|
||||||
|
readonly bool isCompleted;
|
||||||
|
|
||||||
|
public TestAwaiter(bool isCompleted, UniTaskStatus status)
|
||||||
|
{
|
||||||
|
this.isCompleted = isCompleted;
|
||||||
|
this.status = status;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TestAwaiter GetAwaiter() => this;
|
||||||
|
|
||||||
|
public bool IsCompleted => isCompleted;
|
||||||
|
|
||||||
|
public void GetResult()
|
||||||
|
{
|
||||||
|
switch (status)
|
||||||
|
{
|
||||||
|
case UniTaskStatus.Faulted:
|
||||||
|
throw new TaskTestException();
|
||||||
|
case UniTaskStatus.Canceled:
|
||||||
|
throw new OperationCanceledException();
|
||||||
|
case UniTaskStatus.Pending:
|
||||||
|
case UniTaskStatus.Succeeded:
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.QueueUserWorkItem(_ => continuation(), null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.UnsafeQueueUserWorkItem(_ => continuation(), null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct TestAwaiter<T> : ICriticalNotifyCompletion
|
||||||
|
{
|
||||||
|
readonly UniTaskStatus status;
|
||||||
|
readonly bool isCompleted;
|
||||||
|
readonly T value;
|
||||||
|
|
||||||
|
public TestAwaiter(bool isCompleted, UniTaskStatus status, T value)
|
||||||
|
{
|
||||||
|
this.isCompleted = isCompleted;
|
||||||
|
this.status = status;
|
||||||
|
this.value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TestAwaiter<T> GetAwaiter() => this;
|
||||||
|
|
||||||
|
public bool IsCompleted => isCompleted;
|
||||||
|
|
||||||
|
public T GetResult()
|
||||||
|
{
|
||||||
|
switch (status)
|
||||||
|
{
|
||||||
|
case UniTaskStatus.Faulted:
|
||||||
|
throw new TaskTestException();
|
||||||
|
case UniTaskStatus.Canceled:
|
||||||
|
throw new OperationCanceledException();
|
||||||
|
case UniTaskStatus.Pending:
|
||||||
|
case UniTaskStatus.Succeeded:
|
||||||
|
default:
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.QueueUserWorkItem(_ => continuation(), null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
ThreadPool.UnsafeQueueUserWorkItem(_ => continuation(), null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
637
src/UniTask.NetCoreTests/TriggerEventTest.cs
Normal file
637
src/UniTask.NetCoreTests/TriggerEventTest.cs
Normal file
@@ -0,0 +1,637 @@
|
|||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using FluentAssertions;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Channels;
|
||||||
|
using Cysharp.Threading.Tasks.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace NetCoreTests
|
||||||
|
{
|
||||||
|
public class TriggerEventTest
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public void SimpleAdd()
|
||||||
|
{
|
||||||
|
var ev = new TriggerEvent<int>();
|
||||||
|
|
||||||
|
// do nothing
|
||||||
|
ev.SetResult(0);
|
||||||
|
ev.SetError(null);
|
||||||
|
ev.SetCompleted();
|
||||||
|
ev.SetCanceled(default);
|
||||||
|
|
||||||
|
{
|
||||||
|
var one = new TestEvent(1);
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
|
||||||
|
ev.SetCompleted();
|
||||||
|
|
||||||
|
one.CompletedCalled.Count.Should().Be(1);
|
||||||
|
|
||||||
|
// do nothing
|
||||||
|
ev.SetResult(0);
|
||||||
|
ev.SetError(null);
|
||||||
|
ev.SetCompleted();
|
||||||
|
ev.SetCanceled(default);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
one.CompletedCalled.Count.Should().Be(1);
|
||||||
|
}
|
||||||
|
// after removed, onemore
|
||||||
|
{
|
||||||
|
var one = new TestEvent(1);
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
|
||||||
|
ev.SetCompleted();
|
||||||
|
|
||||||
|
one.CompletedCalled.Count.Should().Be(1);
|
||||||
|
|
||||||
|
// do nothing
|
||||||
|
ev.SetResult(0);
|
||||||
|
ev.SetError(null);
|
||||||
|
ev.SetCompleted();
|
||||||
|
ev.SetCanceled(default);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
one.CompletedCalled.Count.Should().Be(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void AddFour()
|
||||||
|
{
|
||||||
|
var ev = new TriggerEvent<int>();
|
||||||
|
|
||||||
|
// do nothing
|
||||||
|
ev.SetResult(0);
|
||||||
|
ev.SetError(null);
|
||||||
|
ev.SetCompleted();
|
||||||
|
ev.SetCanceled(default);
|
||||||
|
|
||||||
|
{
|
||||||
|
var one = new TestEvent(1);
|
||||||
|
var two = new TestEvent(2);
|
||||||
|
var three = new TestEvent(3);
|
||||||
|
var four = new TestEvent(4);
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(three);
|
||||||
|
ev.Add(four);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
four.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
|
||||||
|
ev.SetCompleted();
|
||||||
|
|
||||||
|
one.CompletedCalled.Count.Should().Be(1);
|
||||||
|
two.CompletedCalled.Count.Should().Be(1);
|
||||||
|
three.CompletedCalled.Count.Should().Be(1);
|
||||||
|
|
||||||
|
|
||||||
|
// do nothing
|
||||||
|
ev.SetResult(0);
|
||||||
|
ev.SetError(null);
|
||||||
|
ev.SetCompleted();
|
||||||
|
ev.SetCanceled(default);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
one.CompletedCalled.Count.Should().Be(1);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.CompletedCalled.Count.Should().Be(1);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.CompletedCalled.Count.Should().Be(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// after removed, onemore.
|
||||||
|
{
|
||||||
|
var one = new TestEvent(1);
|
||||||
|
var two = new TestEvent(2);
|
||||||
|
var three = new TestEvent(3);
|
||||||
|
var four = new TestEvent(4);
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(three);
|
||||||
|
ev.Add(four);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
ev.Add(four);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
four.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
|
||||||
|
ev.SetCompleted();
|
||||||
|
|
||||||
|
one.CompletedCalled.Count.Should().Be(1);
|
||||||
|
two.CompletedCalled.Count.Should().Be(1);
|
||||||
|
three.CompletedCalled.Count.Should().Be(1);
|
||||||
|
|
||||||
|
|
||||||
|
// do nothing
|
||||||
|
ev.SetResult(0);
|
||||||
|
ev.SetError(null);
|
||||||
|
ev.SetCompleted();
|
||||||
|
ev.SetCanceled(default);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
one.CompletedCalled.Count.Should().Be(1);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.CompletedCalled.Count.Should().Be(1);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.CompletedCalled.Count.Should().Be(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void OneRemove()
|
||||||
|
{
|
||||||
|
var ev = new TriggerEvent<int>();
|
||||||
|
{
|
||||||
|
var one = new TestEvent(1);
|
||||||
|
var two = new TestEvent(2);
|
||||||
|
var three = new TestEvent(3);
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(three);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
|
||||||
|
ev.Remove(one);
|
||||||
|
|
||||||
|
ev.SetResult(40);
|
||||||
|
ev.SetResult(50);
|
||||||
|
ev.SetResult(60);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30, 40, 50, 60);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30, 40, 50, 60);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
[Fact]
|
||||||
|
public void TwoRemove()
|
||||||
|
{
|
||||||
|
var ev = new TriggerEvent<int>();
|
||||||
|
{
|
||||||
|
var one = new TestEvent(1);
|
||||||
|
var two = new TestEvent(2);
|
||||||
|
var three = new TestEvent(3);
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(three);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
|
||||||
|
ev.Remove(two);
|
||||||
|
|
||||||
|
ev.SetResult(40);
|
||||||
|
ev.SetResult(50);
|
||||||
|
ev.SetResult(60);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30, 40, 50, 60);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30, 40, 50, 60);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
[Fact]
|
||||||
|
public void ThreeRemove()
|
||||||
|
{
|
||||||
|
var ev = new TriggerEvent<int>();
|
||||||
|
{
|
||||||
|
var one = new TestEvent(1);
|
||||||
|
var two = new TestEvent(2);
|
||||||
|
var three = new TestEvent(3);
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(three);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
|
||||||
|
ev.Remove(three);
|
||||||
|
|
||||||
|
ev.SetResult(40);
|
||||||
|
ev.SetResult(50);
|
||||||
|
ev.SetResult(60);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30, 40, 50, 60);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30, 40, 50, 60);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void RemoveSelf()
|
||||||
|
{
|
||||||
|
new RemoveMe().Run1();
|
||||||
|
new RemoveMe().Run2();
|
||||||
|
new RemoveMe().Run3();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void RemoveNextInIterating()
|
||||||
|
{
|
||||||
|
new RemoveNext().Run1();
|
||||||
|
new RemoveNext().Run2();
|
||||||
|
new RemoveNext().Run3();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void RemoveNextNextTest()
|
||||||
|
{
|
||||||
|
new RemoveNextNext().Run1();
|
||||||
|
new RemoveNextNext().Run2();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void AddTest()
|
||||||
|
{
|
||||||
|
new AddMe().Run1();
|
||||||
|
new AddMe().Run2();
|
||||||
|
}
|
||||||
|
|
||||||
|
public class RemoveMe
|
||||||
|
{
|
||||||
|
TriggerEvent<int> ev;
|
||||||
|
|
||||||
|
public void Run1()
|
||||||
|
{
|
||||||
|
TestEvent one = default;
|
||||||
|
one = new TestEvent(1, () => ev.Remove(one));
|
||||||
|
|
||||||
|
var two = new TestEvent(2);
|
||||||
|
var three = new TestEvent(3);
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(three);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Run2()
|
||||||
|
{
|
||||||
|
TestEvent one = default;
|
||||||
|
one = new TestEvent(1, () => ev.Remove(one));
|
||||||
|
|
||||||
|
var two = new TestEvent(2);
|
||||||
|
var three = new TestEvent(3);
|
||||||
|
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(one); // add second.
|
||||||
|
ev.Add(three);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Run3()
|
||||||
|
{
|
||||||
|
TestEvent one = default;
|
||||||
|
one = new TestEvent(1, () => ev.Remove(one));
|
||||||
|
|
||||||
|
var two = new TestEvent(2);
|
||||||
|
var three = new TestEvent(3);
|
||||||
|
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(three);
|
||||||
|
ev.Add(one); // add thired.
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class RemoveNext
|
||||||
|
{
|
||||||
|
TriggerEvent<int> ev;
|
||||||
|
|
||||||
|
public void Run1()
|
||||||
|
{
|
||||||
|
TestEvent one = default;
|
||||||
|
TestEvent two = default;
|
||||||
|
TestEvent three = default;
|
||||||
|
one = new TestEvent(1, () => ev.Remove(two));
|
||||||
|
two = new TestEvent(2);
|
||||||
|
three = new TestEvent(3);
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(three);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
two.NextCalled.Count.Should().Be(0);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Run2()
|
||||||
|
{
|
||||||
|
TestEvent one = default;
|
||||||
|
TestEvent two = default;
|
||||||
|
TestEvent three = default;
|
||||||
|
one = new TestEvent(1, () => ev.Remove(two));
|
||||||
|
two = new TestEvent(2);
|
||||||
|
three = new TestEvent(3);
|
||||||
|
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(one); // add second
|
||||||
|
ev.Add(three);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Run3()
|
||||||
|
{
|
||||||
|
TestEvent one = default;
|
||||||
|
TestEvent two = default;
|
||||||
|
TestEvent three = default;
|
||||||
|
one = new TestEvent(1, () => ev.Remove(two));
|
||||||
|
two = new TestEvent(2);
|
||||||
|
three = new TestEvent(3);
|
||||||
|
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(three);
|
||||||
|
ev.Add(one); // add thired.
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(10);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class RemoveNextNext
|
||||||
|
{
|
||||||
|
TriggerEvent<int> ev;
|
||||||
|
|
||||||
|
public void Run1()
|
||||||
|
{
|
||||||
|
TestEvent one = default;
|
||||||
|
TestEvent two = default;
|
||||||
|
TestEvent three = default;
|
||||||
|
TestEvent four = default;
|
||||||
|
one = new TestEvent(1, () => { ev.Remove(two); ev.Remove(three); });
|
||||||
|
two = new TestEvent(2);
|
||||||
|
three = new TestEvent(3);
|
||||||
|
four = new TestEvent(4);
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(three);
|
||||||
|
ev.Add(four);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
two.NextCalled.Count.Should().Be(0);
|
||||||
|
three.NextCalled.Count.Should().Be(0);
|
||||||
|
four.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Run2()
|
||||||
|
{
|
||||||
|
TestEvent one = default;
|
||||||
|
TestEvent two = default;
|
||||||
|
TestEvent three = default;
|
||||||
|
TestEvent four = default;
|
||||||
|
one = new TestEvent(1, () => { ev.Remove(one); ev.Remove(two); ev.Remove(three); });
|
||||||
|
two = new TestEvent(2);
|
||||||
|
three = new TestEvent(3);
|
||||||
|
four = new TestEvent(4);
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
ev.Add(two);
|
||||||
|
ev.Add(three);
|
||||||
|
ev.Add(four);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10);
|
||||||
|
two.NextCalled.Count.Should().Be(0);
|
||||||
|
three.NextCalled.Count.Should().Be(0);
|
||||||
|
four.NextCalled.Should().BeEquivalentTo(10, 20, 30);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public class AddMe
|
||||||
|
{
|
||||||
|
TriggerEvent<int> ev;
|
||||||
|
|
||||||
|
public void Run1()
|
||||||
|
{
|
||||||
|
TestEvent one = default;
|
||||||
|
TestEvent two = default;
|
||||||
|
TestEvent three = default;
|
||||||
|
TestEvent four = default;
|
||||||
|
|
||||||
|
one = new TestEvent(1, () =>
|
||||||
|
{
|
||||||
|
if (two == null)
|
||||||
|
{
|
||||||
|
ev.Add(two = new TestEvent(2));
|
||||||
|
}
|
||||||
|
else if (three == null)
|
||||||
|
{
|
||||||
|
ev.Add(three = new TestEvent(3));
|
||||||
|
}
|
||||||
|
else if (four == null)
|
||||||
|
{
|
||||||
|
ev.Add(four = new TestEvent(4));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
ev.SetResult(40);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30, 40);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(20, 30, 40);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(30, 40);
|
||||||
|
four.NextCalled.Should().BeEquivalentTo(40);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Run2()
|
||||||
|
{
|
||||||
|
TestEvent one = default;
|
||||||
|
TestEvent two = default;
|
||||||
|
TestEvent three = default;
|
||||||
|
TestEvent four = default;
|
||||||
|
|
||||||
|
one = new TestEvent(1, () =>
|
||||||
|
{
|
||||||
|
if (two == null)
|
||||||
|
{
|
||||||
|
ev.Add(two = new TestEvent(2, () =>
|
||||||
|
{
|
||||||
|
if (three == null)
|
||||||
|
{
|
||||||
|
ev.Add(three = new TestEvent(3, () =>
|
||||||
|
{
|
||||||
|
if (four == null)
|
||||||
|
{
|
||||||
|
ev.Add(four = new TestEvent(4));
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
ev.Add(one);
|
||||||
|
|
||||||
|
ev.SetResult(10);
|
||||||
|
ev.SetResult(20);
|
||||||
|
ev.SetResult(30);
|
||||||
|
ev.SetResult(40);
|
||||||
|
|
||||||
|
one.NextCalled.Should().BeEquivalentTo(10, 20, 30, 40);
|
||||||
|
two.NextCalled.Should().BeEquivalentTo(20, 30, 40);
|
||||||
|
three.NextCalled.Should().BeEquivalentTo(30, 40);
|
||||||
|
four.NextCalled.Should().BeEquivalentTo(40);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class TestEvent : ITriggerHandler<int>
|
||||||
|
{
|
||||||
|
public readonly int Id;
|
||||||
|
readonly Action iteratingEvent;
|
||||||
|
|
||||||
|
public TestEvent(int id)
|
||||||
|
{
|
||||||
|
this.Id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TestEvent(int id, Action iteratingEvent)
|
||||||
|
{
|
||||||
|
this.Id = id;
|
||||||
|
this.iteratingEvent = iteratingEvent;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<int> NextCalled = new List<int>();
|
||||||
|
public List<Exception> ErrorCalled = new List<Exception>();
|
||||||
|
public List<object> CompletedCalled = new List<object>();
|
||||||
|
public List<CancellationToken> CancelCalled = new List<CancellationToken>();
|
||||||
|
|
||||||
|
public ITriggerHandler<int> Prev { get; set; }
|
||||||
|
public ITriggerHandler<int> Next { get; set; }
|
||||||
|
|
||||||
|
public void OnCanceled(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
CancelCalled.Add(cancellationToken);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted()
|
||||||
|
{
|
||||||
|
CompletedCalled.Add(new object());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnError(Exception ex)
|
||||||
|
{
|
||||||
|
ErrorCalled.Add(ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnNext(int value)
|
||||||
|
{
|
||||||
|
NextCalled.Add(value);
|
||||||
|
iteratingEvent?.Invoke();
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
return Id.ToString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
30
src/UniTask/Assets/Editor/EditorRunnerChecker.cs
Normal file
30
src/UniTask/Assets/Editor/EditorRunnerChecker.cs
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
#if UNITY_EDITOR
|
||||||
|
|
||||||
|
using Cysharp.Threading.Tasks;
|
||||||
|
using System;
|
||||||
|
using System.IO;
|
||||||
|
using System.Linq;
|
||||||
|
using UnityEditor;
|
||||||
|
using UnityEngine;
|
||||||
|
using UnityEngine.Networking;
|
||||||
|
|
||||||
|
public static class EditorRunnerChecker
|
||||||
|
{
|
||||||
|
[MenuItem("Tools/UniTaskEditorRunnerChecker")]
|
||||||
|
public static void RunUniTaskAsync()
|
||||||
|
{
|
||||||
|
RunCore().Forget();
|
||||||
|
}
|
||||||
|
|
||||||
|
static async UniTaskVoid RunCore()
|
||||||
|
{
|
||||||
|
Debug.Log("Start");
|
||||||
|
|
||||||
|
var r = await UnityWebRequest.Get("https://bing.com/").SendWebRequest().ToUniTask();
|
||||||
|
Debug.Log(r.downloadHandler.text.Substring(0, 100));
|
||||||
|
|
||||||
|
Debug.Log("End");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
fileFormatVersion: 2
|
fileFormatVersion: 2
|
||||||
guid: fcb1f7467a3e2b64c8a016c8aee2f9b4
|
guid: e51b78c06cb410f42b36e0af9de3b065
|
||||||
MonoImporter:
|
MonoImporter:
|
||||||
externalObjects: {}
|
externalObjects: {}
|
||||||
serializedVersion: 2
|
serializedVersion: 2
|
||||||
@@ -0,0 +1,375 @@
|
|||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks
|
||||||
|
{
|
||||||
|
public interface IReadOnlyAsyncReactiveProperty<T> : IUniTaskAsyncEnumerable<T>
|
||||||
|
{
|
||||||
|
T Value { get; }
|
||||||
|
IUniTaskAsyncEnumerable<T> WithoutCurrent();
|
||||||
|
}
|
||||||
|
|
||||||
|
public interface IAsyncReactiveProperty<T> : IReadOnlyAsyncReactiveProperty<T>
|
||||||
|
{
|
||||||
|
new T Value { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
[Serializable]
|
||||||
|
public class AsyncReactiveProperty<T> : IAsyncReactiveProperty<T>, IDisposable
|
||||||
|
{
|
||||||
|
TriggerEvent<T> triggerEvent;
|
||||||
|
|
||||||
|
#if UNITY_2018_3_OR_NEWER
|
||||||
|
[UnityEngine.SerializeField]
|
||||||
|
#endif
|
||||||
|
T latestValue;
|
||||||
|
|
||||||
|
public T Value
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
return latestValue;
|
||||||
|
}
|
||||||
|
set
|
||||||
|
{
|
||||||
|
this.latestValue = value;
|
||||||
|
triggerEvent.SetResult(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public AsyncReactiveProperty(T value)
|
||||||
|
{
|
||||||
|
this.latestValue = value;
|
||||||
|
this.triggerEvent = default;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IUniTaskAsyncEnumerable<T> WithoutCurrent()
|
||||||
|
{
|
||||||
|
return new WithoutCurrentEnumerable(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
return new Enumerator(this, cancellationToken, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
triggerEvent.SetCompleted();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static implicit operator T(AsyncReactiveProperty<T> value)
|
||||||
|
{
|
||||||
|
return value.Value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
if (isValueType) return latestValue.ToString();
|
||||||
|
return latestValue?.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool isValueType;
|
||||||
|
|
||||||
|
static AsyncReactiveProperty()
|
||||||
|
{
|
||||||
|
isValueType = typeof(T).IsValueType;
|
||||||
|
}
|
||||||
|
|
||||||
|
class WithoutCurrentEnumerable : IUniTaskAsyncEnumerable<T>
|
||||||
|
{
|
||||||
|
readonly AsyncReactiveProperty<T> parent;
|
||||||
|
|
||||||
|
public WithoutCurrentEnumerable(AsyncReactiveProperty<T> parent)
|
||||||
|
{
|
||||||
|
this.parent = parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
return new Enumerator(parent, cancellationToken, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<T>, ITriggerHandler<T>
|
||||||
|
{
|
||||||
|
static Action<object> cancellationCallback = CancellationCallback;
|
||||||
|
|
||||||
|
readonly AsyncReactiveProperty<T> parent;
|
||||||
|
readonly CancellationToken cancellationToken;
|
||||||
|
readonly CancellationTokenRegistration cancellationTokenRegistration;
|
||||||
|
T value;
|
||||||
|
bool isDisposed;
|
||||||
|
bool firstCall;
|
||||||
|
|
||||||
|
public Enumerator(AsyncReactiveProperty<T> parent, CancellationToken cancellationToken, bool publishCurrentValue)
|
||||||
|
{
|
||||||
|
this.parent = parent;
|
||||||
|
this.cancellationToken = cancellationToken;
|
||||||
|
this.firstCall = publishCurrentValue;
|
||||||
|
|
||||||
|
parent.triggerEvent.Add(this);
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
|
|
||||||
|
if (cancellationToken.CanBeCanceled)
|
||||||
|
{
|
||||||
|
cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(cancellationCallback, this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public T Current => value;
|
||||||
|
|
||||||
|
ITriggerHandler<T> ITriggerHandler<T>.Prev { get; set; }
|
||||||
|
ITriggerHandler<T> ITriggerHandler<T>.Next { get; set; }
|
||||||
|
|
||||||
|
public UniTask<bool> MoveNextAsync()
|
||||||
|
{
|
||||||
|
// raise latest value on first call.
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
firstCall = false;
|
||||||
|
value = parent.Value;
|
||||||
|
return CompletedTasks.True;
|
||||||
|
}
|
||||||
|
|
||||||
|
completionSource.Reset();
|
||||||
|
return new UniTask<bool>(this, completionSource.Version);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask DisposeAsync()
|
||||||
|
{
|
||||||
|
if (!isDisposed)
|
||||||
|
{
|
||||||
|
isDisposed = true;
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
completionSource.TrySetCanceled(cancellationToken);
|
||||||
|
parent.triggerEvent.Remove(this);
|
||||||
|
}
|
||||||
|
return default;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnNext(T value)
|
||||||
|
{
|
||||||
|
this.value = value;
|
||||||
|
completionSource.TrySetResult(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCanceled(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
DisposeAsync().Forget();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted()
|
||||||
|
{
|
||||||
|
completionSource.TrySetResult(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnError(Exception ex)
|
||||||
|
{
|
||||||
|
completionSource.TrySetException(ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void CancellationCallback(object state)
|
||||||
|
{
|
||||||
|
var self = (Enumerator)state;
|
||||||
|
self.DisposeAsync().Forget();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class ReadOnlyAsyncReactiveProperty<T> : IReadOnlyAsyncReactiveProperty<T>, IDisposable
|
||||||
|
{
|
||||||
|
TriggerEvent<T> triggerEvent;
|
||||||
|
|
||||||
|
T latestValue;
|
||||||
|
IUniTaskAsyncEnumerator<T> enumerator;
|
||||||
|
|
||||||
|
public T Value
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
return latestValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public ReadOnlyAsyncReactiveProperty(T initialValue, IUniTaskAsyncEnumerable<T> source, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
latestValue = initialValue;
|
||||||
|
ConsumeEnumerator(source, cancellationToken).Forget();
|
||||||
|
}
|
||||||
|
|
||||||
|
public ReadOnlyAsyncReactiveProperty(IUniTaskAsyncEnumerable<T> source, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
ConsumeEnumerator(source, cancellationToken).Forget();
|
||||||
|
}
|
||||||
|
|
||||||
|
async UniTaskVoid ConsumeEnumerator(IUniTaskAsyncEnumerable<T> source, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
enumerator = source.GetAsyncEnumerator(cancellationToken);
|
||||||
|
try
|
||||||
|
{
|
||||||
|
while (await enumerator.MoveNextAsync())
|
||||||
|
{
|
||||||
|
var value = enumerator.Current;
|
||||||
|
this.latestValue = value;
|
||||||
|
triggerEvent.SetResult(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
await enumerator.DisposeAsync();
|
||||||
|
enumerator = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public IUniTaskAsyncEnumerable<T> WithoutCurrent()
|
||||||
|
{
|
||||||
|
return new WithoutCurrentEnumerable(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
return new Enumerator(this, cancellationToken, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
if (enumerator != null)
|
||||||
|
{
|
||||||
|
enumerator.DisposeAsync().Forget();
|
||||||
|
}
|
||||||
|
|
||||||
|
triggerEvent.SetCompleted();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static implicit operator T(ReadOnlyAsyncReactiveProperty<T> value)
|
||||||
|
{
|
||||||
|
return value.Value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
if (isValueType) return latestValue.ToString();
|
||||||
|
return latestValue?.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool isValueType;
|
||||||
|
|
||||||
|
static ReadOnlyAsyncReactiveProperty()
|
||||||
|
{
|
||||||
|
isValueType = typeof(T).IsValueType;
|
||||||
|
}
|
||||||
|
|
||||||
|
class WithoutCurrentEnumerable : IUniTaskAsyncEnumerable<T>
|
||||||
|
{
|
||||||
|
readonly ReadOnlyAsyncReactiveProperty<T> parent;
|
||||||
|
|
||||||
|
public WithoutCurrentEnumerable(ReadOnlyAsyncReactiveProperty<T> parent)
|
||||||
|
{
|
||||||
|
this.parent = parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
return new Enumerator(parent, cancellationToken, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<T>, ITriggerHandler<T>
|
||||||
|
{
|
||||||
|
static Action<object> cancellationCallback = CancellationCallback;
|
||||||
|
|
||||||
|
readonly ReadOnlyAsyncReactiveProperty<T> parent;
|
||||||
|
readonly CancellationToken cancellationToken;
|
||||||
|
readonly CancellationTokenRegistration cancellationTokenRegistration;
|
||||||
|
T value;
|
||||||
|
bool isDisposed;
|
||||||
|
bool firstCall;
|
||||||
|
|
||||||
|
public Enumerator(ReadOnlyAsyncReactiveProperty<T> parent, CancellationToken cancellationToken, bool publishCurrentValue)
|
||||||
|
{
|
||||||
|
this.parent = parent;
|
||||||
|
this.cancellationToken = cancellationToken;
|
||||||
|
this.firstCall = publishCurrentValue;
|
||||||
|
|
||||||
|
parent.triggerEvent.Add(this);
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
|
|
||||||
|
if (cancellationToken.CanBeCanceled)
|
||||||
|
{
|
||||||
|
cancellationTokenRegistration = cancellationToken.RegisterWithoutCaptureExecutionContext(cancellationCallback, this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public T Current => value;
|
||||||
|
ITriggerHandler<T> ITriggerHandler<T>.Prev { get; set; }
|
||||||
|
ITriggerHandler<T> ITriggerHandler<T>.Next { get; set; }
|
||||||
|
|
||||||
|
public UniTask<bool> MoveNextAsync()
|
||||||
|
{
|
||||||
|
// raise latest value on first call.
|
||||||
|
if (firstCall)
|
||||||
|
{
|
||||||
|
firstCall = false;
|
||||||
|
value = parent.Value;
|
||||||
|
return CompletedTasks.True;
|
||||||
|
}
|
||||||
|
|
||||||
|
completionSource.Reset();
|
||||||
|
return new UniTask<bool>(this, completionSource.Version);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask DisposeAsync()
|
||||||
|
{
|
||||||
|
if (!isDisposed)
|
||||||
|
{
|
||||||
|
isDisposed = true;
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
completionSource.TrySetCanceled(cancellationToken);
|
||||||
|
parent.triggerEvent.Remove(this);
|
||||||
|
}
|
||||||
|
return default;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnNext(T value)
|
||||||
|
{
|
||||||
|
this.value = value;
|
||||||
|
completionSource.TrySetResult(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCanceled(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
DisposeAsync().Forget();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted()
|
||||||
|
{
|
||||||
|
completionSource.TrySetResult(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnError(Exception ex)
|
||||||
|
{
|
||||||
|
completionSource.TrySetException(ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void CancellationCallback(object state)
|
||||||
|
{
|
||||||
|
var self = (Enumerator)state;
|
||||||
|
self.DisposeAsync().Forget();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class StateExtensions
|
||||||
|
{
|
||||||
|
public static ReadOnlyAsyncReactiveProperty<T> ToReadOnlyAsyncReactiveProperty<T>(this IUniTaskAsyncEnumerable<T> source, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
return new ReadOnlyAsyncReactiveProperty<T>(source, cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ReadOnlyAsyncReactiveProperty<T> ToReadOnlyAsyncReactiveProperty<T>(this IUniTaskAsyncEnumerable<T> source, T initialValue, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
return new ReadOnlyAsyncReactiveProperty<T>(initialValue, source, cancellationToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
fileFormatVersion: 2
|
fileFormatVersion: 2
|
||||||
guid: 4d5a9a3e1f0f069478969f752fde29a9
|
guid: 8ef320b87f537ee4fb2282e765dc6166
|
||||||
MonoImporter:
|
MonoImporter:
|
||||||
externalObjects: {}
|
externalObjects: {}
|
||||||
serializedVersion: 2
|
serializedVersion: 2
|
||||||
@@ -4,7 +4,7 @@ using System;
|
|||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
namespace Cysharp.Threading.Tasks
|
||||||
{
|
{
|
||||||
public struct AsyncUnit : IEquatable<AsyncUnit>
|
public readonly struct AsyncUnit : IEquatable<AsyncUnit>
|
||||||
{
|
{
|
||||||
public static readonly AsyncUnit Default = new AsyncUnit();
|
public static readonly AsyncUnit Default = new AsyncUnit();
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks
|
namespace Cysharp.Threading.Tasks
|
||||||
@@ -9,15 +10,15 @@ namespace Cysharp.Threading.Tasks
|
|||||||
{
|
{
|
||||||
static readonly Action<object> cancellationTokenCallback = Callback;
|
static readonly Action<object> cancellationTokenCallback = Callback;
|
||||||
|
|
||||||
public static (UniTask, CancellationTokenRegistration) ToUniTask(this CancellationToken cts)
|
public static (UniTask, CancellationTokenRegistration) ToUniTask(this CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
if (cts.IsCancellationRequested)
|
if (cancellationToken.IsCancellationRequested)
|
||||||
{
|
{
|
||||||
return (UniTask.FromCanceled(cts), default(CancellationTokenRegistration));
|
return (UniTask.FromCanceled(cancellationToken), default(CancellationTokenRegistration));
|
||||||
}
|
}
|
||||||
|
|
||||||
var promise = new UniTaskCompletionSource();
|
var promise = new UniTaskCompletionSource();
|
||||||
return (promise.Task, cts.RegisterWithoutCaptureExecutionContext(cancellationTokenCallback, promise));
|
return (promise.Task, cancellationToken.RegisterWithoutCaptureExecutionContext(cancellationTokenCallback, promise));
|
||||||
}
|
}
|
||||||
|
|
||||||
static void Callback(object state)
|
static void Callback(object state)
|
||||||
@@ -26,6 +27,11 @@ namespace Cysharp.Threading.Tasks
|
|||||||
promise.TrySetResult();
|
promise.TrySetResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static CancellationTokenAwaitable WaitUntilCanceled(this CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
return new CancellationTokenAwaitable(cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
public static CancellationTokenRegistration RegisterWithoutCaptureExecutionContext(this CancellationToken cancellationToken, Action callback)
|
public static CancellationTokenRegistration RegisterWithoutCaptureExecutionContext(this CancellationToken cancellationToken, Action callback)
|
||||||
{
|
{
|
||||||
var restoreFlow = false;
|
var restoreFlow = false;
|
||||||
@@ -70,5 +76,46 @@ namespace Cysharp.Threading.Tasks
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public struct CancellationTokenAwaitable
|
||||||
|
{
|
||||||
|
CancellationToken cancellationToken;
|
||||||
|
|
||||||
|
public CancellationTokenAwaitable(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
this.cancellationToken = cancellationToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Awaiter GetAwaiter()
|
||||||
|
{
|
||||||
|
return new Awaiter(cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct Awaiter : ICriticalNotifyCompletion
|
||||||
|
{
|
||||||
|
CancellationToken cancellationToken;
|
||||||
|
|
||||||
|
public Awaiter(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
this.cancellationToken = cancellationToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool IsCompleted => !cancellationToken.CanBeCanceled || cancellationToken.IsCancellationRequested;
|
||||||
|
|
||||||
|
public void GetResult()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
UnsafeOnCompleted(continuation);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
|
{
|
||||||
|
cancellationToken.RegisterWithoutCaptureExecutionContext(continuation);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
450
src/UniTask/Assets/Plugins/UniTask/Runtime/Channel.cs
Normal file
450
src/UniTask/Assets/Plugins/UniTask/Runtime/Channel.cs
Normal file
@@ -0,0 +1,450 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks
|
||||||
|
{
|
||||||
|
public static class Channel
|
||||||
|
{
|
||||||
|
public static Channel<T> CreateSingleConsumerUnbounded<T>()
|
||||||
|
{
|
||||||
|
return new SingleConsumerUnboundedChannel<T>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public abstract class Channel<TWrite, TRead>
|
||||||
|
{
|
||||||
|
public ChannelReader<TRead> Reader { get; protected set; }
|
||||||
|
public ChannelWriter<TWrite> Writer { get; protected set; }
|
||||||
|
|
||||||
|
public static implicit operator ChannelReader<TRead>(Channel<TWrite, TRead> channel) => channel.Reader;
|
||||||
|
public static implicit operator ChannelWriter<TWrite>(Channel<TWrite, TRead> channel) => channel.Writer;
|
||||||
|
}
|
||||||
|
|
||||||
|
public abstract class Channel<T> : Channel<T, T>
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public abstract class ChannelReader<T>
|
||||||
|
{
|
||||||
|
public abstract bool TryRead(out T item);
|
||||||
|
public abstract UniTask<bool> WaitToReadAsync(CancellationToken cancellationToken = default(CancellationToken));
|
||||||
|
|
||||||
|
public abstract UniTask Completion { get; }
|
||||||
|
|
||||||
|
public virtual UniTask<T> ReadAsync(CancellationToken cancellationToken = default(CancellationToken))
|
||||||
|
{
|
||||||
|
if (this.TryRead(out var item))
|
||||||
|
{
|
||||||
|
return UniTask.FromResult(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ReadAsyncCore(cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
async UniTask<T> ReadAsyncCore(CancellationToken cancellationToken = default(CancellationToken))
|
||||||
|
{
|
||||||
|
if (await WaitToReadAsync(cancellationToken))
|
||||||
|
{
|
||||||
|
if (TryRead(out var item))
|
||||||
|
{
|
||||||
|
return item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new ChannelClosedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
public abstract IUniTaskAsyncEnumerable<T> ReadAllAsync(CancellationToken cancellationToken = default(CancellationToken));
|
||||||
|
}
|
||||||
|
|
||||||
|
public abstract class ChannelWriter<T>
|
||||||
|
{
|
||||||
|
public abstract bool TryWrite(T item);
|
||||||
|
public abstract bool TryComplete(Exception error = null);
|
||||||
|
|
||||||
|
public void Complete(Exception error = null)
|
||||||
|
{
|
||||||
|
if (!TryComplete(error))
|
||||||
|
{
|
||||||
|
throw new ChannelClosedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public partial class ChannelClosedException : InvalidOperationException
|
||||||
|
{
|
||||||
|
public ChannelClosedException() :
|
||||||
|
base("Channel is already closed.")
|
||||||
|
{ }
|
||||||
|
|
||||||
|
public ChannelClosedException(string message) : base(message) { }
|
||||||
|
|
||||||
|
public ChannelClosedException(Exception innerException) :
|
||||||
|
base("Channel is already closed", innerException)
|
||||||
|
{ }
|
||||||
|
|
||||||
|
public ChannelClosedException(string message, Exception innerException) : base(message, innerException) { }
|
||||||
|
}
|
||||||
|
|
||||||
|
internal class SingleConsumerUnboundedChannel<T> : Channel<T>
|
||||||
|
{
|
||||||
|
readonly Queue<T> items;
|
||||||
|
readonly SingleConsumerUnboundedChannelReader readerSource;
|
||||||
|
UniTaskCompletionSource completedTaskSource;
|
||||||
|
UniTask completedTask;
|
||||||
|
|
||||||
|
Exception completionError;
|
||||||
|
bool closed;
|
||||||
|
|
||||||
|
public SingleConsumerUnboundedChannel()
|
||||||
|
{
|
||||||
|
items = new Queue<T>();
|
||||||
|
Writer = new SingleConsumerUnboundedChannelWriter(this);
|
||||||
|
readerSource = new SingleConsumerUnboundedChannelReader(this);
|
||||||
|
Reader = readerSource;
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class SingleConsumerUnboundedChannelWriter : ChannelWriter<T>
|
||||||
|
{
|
||||||
|
readonly SingleConsumerUnboundedChannel<T> parent;
|
||||||
|
|
||||||
|
public SingleConsumerUnboundedChannelWriter(SingleConsumerUnboundedChannel<T> parent)
|
||||||
|
{
|
||||||
|
this.parent = parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override bool TryWrite(T item)
|
||||||
|
{
|
||||||
|
bool waiting;
|
||||||
|
lock (parent.items)
|
||||||
|
{
|
||||||
|
if (parent.closed) return false;
|
||||||
|
|
||||||
|
parent.items.Enqueue(item);
|
||||||
|
waiting = parent.readerSource.isWaiting;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (waiting)
|
||||||
|
{
|
||||||
|
parent.readerSource.SingalContinuation();
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override bool TryComplete(Exception error = null)
|
||||||
|
{
|
||||||
|
bool waiting;
|
||||||
|
lock (parent.items)
|
||||||
|
{
|
||||||
|
if (parent.closed) return false;
|
||||||
|
parent.closed = true;
|
||||||
|
waiting = parent.readerSource.isWaiting;
|
||||||
|
|
||||||
|
if (parent.items.Count == 0)
|
||||||
|
{
|
||||||
|
if (error == null)
|
||||||
|
{
|
||||||
|
if (parent.completedTaskSource != null)
|
||||||
|
{
|
||||||
|
parent.completedTaskSource.TrySetResult();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
parent.completedTask = UniTask.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (parent.completedTaskSource != null)
|
||||||
|
{
|
||||||
|
parent.completedTaskSource.TrySetException(error);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
parent.completedTask = UniTask.FromException(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (waiting)
|
||||||
|
{
|
||||||
|
parent.readerSource.SingalCompleted(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
parent.completionError = error;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class SingleConsumerUnboundedChannelReader : ChannelReader<T>, IUniTaskSource<bool>
|
||||||
|
{
|
||||||
|
readonly Action<object> CancellationCallbackDelegate = CancellationCallback;
|
||||||
|
readonly SingleConsumerUnboundedChannel<T> parent;
|
||||||
|
|
||||||
|
CancellationToken cancellationToken;
|
||||||
|
CancellationTokenRegistration cancellationTokenRegistration;
|
||||||
|
UniTaskCompletionSourceCore<bool> core;
|
||||||
|
internal bool isWaiting;
|
||||||
|
|
||||||
|
public SingleConsumerUnboundedChannelReader(SingleConsumerUnboundedChannel<T> parent)
|
||||||
|
{
|
||||||
|
this.parent = parent;
|
||||||
|
|
||||||
|
TaskTracker.TrackActiveTask(this, 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
public override UniTask Completion
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
if (parent.completedTaskSource != null) return parent.completedTaskSource.Task;
|
||||||
|
|
||||||
|
if (parent.closed)
|
||||||
|
{
|
||||||
|
return parent.completedTask;
|
||||||
|
}
|
||||||
|
|
||||||
|
parent.completedTaskSource = new UniTaskCompletionSource();
|
||||||
|
return parent.completedTaskSource.Task;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override bool TryRead(out T item)
|
||||||
|
{
|
||||||
|
lock (parent.items)
|
||||||
|
{
|
||||||
|
if (parent.items.Count != 0)
|
||||||
|
{
|
||||||
|
item = parent.items.Dequeue();
|
||||||
|
|
||||||
|
// complete when all value was consumed.
|
||||||
|
if (parent.closed && parent.items.Count == 0)
|
||||||
|
{
|
||||||
|
if (parent.completionError != null)
|
||||||
|
{
|
||||||
|
if (parent.completedTaskSource != null)
|
||||||
|
{
|
||||||
|
parent.completedTaskSource.TrySetException(parent.completionError);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
parent.completedTask = UniTask.FromException(parent.completionError);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (parent.completedTaskSource != null)
|
||||||
|
{
|
||||||
|
parent.completedTaskSource.TrySetResult();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
parent.completedTask = UniTask.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
item = default;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override UniTask<bool> WaitToReadAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
if (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
return UniTask.FromCanceled<bool>(cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
lock (parent.items)
|
||||||
|
{
|
||||||
|
if (parent.items.Count != 0)
|
||||||
|
{
|
||||||
|
return CompletedTasks.True;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parent.closed)
|
||||||
|
{
|
||||||
|
if (parent.completionError == null)
|
||||||
|
{
|
||||||
|
return CompletedTasks.False;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return UniTask.FromException<bool>(parent.completionError);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cancellationTokenRegistration.Dispose();
|
||||||
|
|
||||||
|
core.Reset();
|
||||||
|
isWaiting = true;
|
||||||
|
|
||||||
|
this.cancellationToken = cancellationToken;
|
||||||
|
if (this.cancellationToken.CanBeCanceled)
|
||||||
|
{
|
||||||
|
cancellationTokenRegistration = this.cancellationToken.RegisterWithoutCaptureExecutionContext(CancellationCallbackDelegate, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new UniTask<bool>(this, core.Version);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SingalContinuation()
|
||||||
|
{
|
||||||
|
core.TrySetResult(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SingalCancellation(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
core.TrySetCanceled(cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SingalCompleted(Exception error)
|
||||||
|
{
|
||||||
|
if (error != null)
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
core.TrySetException(error);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
core.TrySetResult(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override IUniTaskAsyncEnumerable<T> ReadAllAsync(CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
return new ReadAllAsyncEnumerable(this, cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IUniTaskSource<bool>.GetResult(short token)
|
||||||
|
{
|
||||||
|
return core.GetResult(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
void IUniTaskSource.GetResult(short token)
|
||||||
|
{
|
||||||
|
core.GetResult(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
UniTaskStatus IUniTaskSource.GetStatus(short token)
|
||||||
|
{
|
||||||
|
return core.GetStatus(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
void IUniTaskSource.OnCompleted(Action<object> continuation, object state, short token)
|
||||||
|
{
|
||||||
|
core.OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
UniTaskStatus IUniTaskSource.UnsafeGetStatus()
|
||||||
|
{
|
||||||
|
return core.UnsafeGetStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
static void CancellationCallback(object state)
|
||||||
|
{
|
||||||
|
var self = (SingleConsumerUnboundedChannelReader)state;
|
||||||
|
self.SingalCancellation(self.cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class ReadAllAsyncEnumerable : IUniTaskAsyncEnumerable<T>, IUniTaskAsyncEnumerator<T>
|
||||||
|
{
|
||||||
|
readonly Action<object> CancellationCallback1Delegate = CancellationCallback1;
|
||||||
|
readonly Action<object> CancellationCallback2Delegate = CancellationCallback2;
|
||||||
|
|
||||||
|
readonly SingleConsumerUnboundedChannelReader parent;
|
||||||
|
CancellationToken cancellationToken1;
|
||||||
|
CancellationToken cancellationToken2;
|
||||||
|
CancellationTokenRegistration cancellationTokenRegistration1;
|
||||||
|
CancellationTokenRegistration cancellationTokenRegistration2;
|
||||||
|
|
||||||
|
T current;
|
||||||
|
bool cacheValue;
|
||||||
|
bool running;
|
||||||
|
|
||||||
|
public ReadAllAsyncEnumerable(SingleConsumerUnboundedChannelReader parent, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
this.parent = parent;
|
||||||
|
this.cancellationToken1 = cancellationToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
if (running)
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException("Enumerator is already running, does not allow call GetAsyncEnumerator twice.");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.cancellationToken1 != cancellationToken)
|
||||||
|
{
|
||||||
|
this.cancellationToken2 = cancellationToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.cancellationToken1.CanBeCanceled)
|
||||||
|
{
|
||||||
|
this.cancellationTokenRegistration1 = this.cancellationToken1.RegisterWithoutCaptureExecutionContext(CancellationCallback1Delegate, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.cancellationToken2.CanBeCanceled)
|
||||||
|
{
|
||||||
|
this.cancellationTokenRegistration2 = this.cancellationToken2.RegisterWithoutCaptureExecutionContext(CancellationCallback2Delegate, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
running = true;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public T Current
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
if (cacheValue)
|
||||||
|
{
|
||||||
|
return current;
|
||||||
|
}
|
||||||
|
parent.TryRead(out current);
|
||||||
|
return current;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask<bool> MoveNextAsync()
|
||||||
|
{
|
||||||
|
cacheValue = false;
|
||||||
|
return parent.WaitToReadAsync(CancellationToken.None); // ok to use None, registered in ctor.
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask DisposeAsync()
|
||||||
|
{
|
||||||
|
cancellationTokenRegistration1.Dispose();
|
||||||
|
cancellationTokenRegistration2.Dispose();
|
||||||
|
return default;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void CancellationCallback1(object state)
|
||||||
|
{
|
||||||
|
var self = (ReadAllAsyncEnumerable)state;
|
||||||
|
self.parent.SingalCancellation(self.cancellationToken1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void CancellationCallback2(object state)
|
||||||
|
{
|
||||||
|
var self = (ReadAllAsyncEnumerable)state;
|
||||||
|
self.parent.SingalCancellation(self.cancellationToken2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
11
src/UniTask/Assets/Plugins/UniTask/Runtime/Channel.cs.meta
Normal file
11
src/UniTask/Assets/Plugins/UniTask/Runtime/Channel.cs.meta
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
fileFormatVersion: 2
|
||||||
|
guid: 5ceb3107bbdd1f14eb39091273798360
|
||||||
|
MonoImporter:
|
||||||
|
externalObjects: {}
|
||||||
|
serializedVersion: 2
|
||||||
|
defaultReferences: []
|
||||||
|
executionOrder: 0
|
||||||
|
icon: {instanceID: 0}
|
||||||
|
userData:
|
||||||
|
assetBundleName:
|
||||||
|
assetBundleVariant:
|
||||||
@@ -12,9 +12,8 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
|||||||
[StructLayout(LayoutKind.Auto)]
|
[StructLayout(LayoutKind.Auto)]
|
||||||
public struct AsyncUniTaskMethodBuilder
|
public struct AsyncUniTaskMethodBuilder
|
||||||
{
|
{
|
||||||
// cache items.
|
IStateMachineRunnerPromise runnerPromise;
|
||||||
AutoResetUniTaskCompletionSource promise;
|
Exception ex;
|
||||||
IMoveNextRunner runner;
|
|
||||||
|
|
||||||
// 1. Static Create method.
|
// 1. Static Create method.
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
@@ -31,98 +30,81 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
|||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
get
|
get
|
||||||
{
|
{
|
||||||
if (promise != null)
|
if (runnerPromise != null)
|
||||||
{
|
{
|
||||||
return promise.Task;
|
return runnerPromise.Task;
|
||||||
}
|
}
|
||||||
|
else if (ex != null)
|
||||||
if (runner == null)
|
{
|
||||||
|
return UniTask.FromException(ex);
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
return UniTask.CompletedTask;
|
return UniTask.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
promise = AutoResetUniTaskCompletionSource.Create();
|
|
||||||
return promise.Task;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. SetException
|
// 3. SetException
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void SetException(Exception exception)
|
public void SetException(Exception exception)
|
||||||
{
|
{
|
||||||
// runner is finished, return first.
|
if (runnerPromise == null)
|
||||||
if (runner != null)
|
|
||||||
{
|
{
|
||||||
runner.Return();
|
ex = exception;
|
||||||
runner = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (promise != null)
|
|
||||||
{
|
|
||||||
promise.TrySetException(exception);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
promise = AutoResetUniTaskCompletionSource.CreateFromException(exception, out _);
|
runnerPromise.SetException(exception);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 4. SetResult
|
// 4. SetResult
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void SetResult()
|
public void SetResult()
|
||||||
{
|
{
|
||||||
// runner is finished, return first.
|
if (runnerPromise != null)
|
||||||
if (runner != null)
|
|
||||||
{
|
{
|
||||||
runner.Return();
|
runnerPromise.SetResult();
|
||||||
runner = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (promise != null)
|
|
||||||
{
|
|
||||||
promise.TrySetResult();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 5. AwaitOnCompleted
|
// 5. AwaitOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : INotifyCompletion
|
where TAwaiter : INotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (promise == null)
|
if (runnerPromise == null)
|
||||||
{
|
{
|
||||||
promise = AutoResetUniTaskCompletionSource.Create();
|
AsyncUniTask<TStateMachine>.SetStateMachine(ref stateMachine, ref runnerPromise);
|
||||||
}
|
|
||||||
if (runner == null)
|
|
||||||
{
|
|
||||||
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.OnCompleted(runner.CallMoveNext);
|
awaiter.OnCompleted(runnerPromise.MoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 6. AwaitUnsafeOnCompleted
|
// 6. AwaitUnsafeOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[SecuritySafeCritical]
|
[SecuritySafeCritical]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : ICriticalNotifyCompletion
|
where TAwaiter : ICriticalNotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (promise == null)
|
if (runnerPromise == null)
|
||||||
{
|
{
|
||||||
promise = AutoResetUniTaskCompletionSource.Create();
|
AsyncUniTask<TStateMachine>.SetStateMachine(ref stateMachine, ref runnerPromise);
|
||||||
}
|
|
||||||
if (runner == null)
|
|
||||||
{
|
|
||||||
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.OnCompleted(runner.CallMoveNext);
|
awaiter.UnsafeOnCompleted(runnerPromise.MoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 7. Start
|
// 7. Start
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void Start<TStateMachine>(ref TStateMachine stateMachine)
|
public void Start<TStateMachine>(ref TStateMachine stateMachine)
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
@@ -156,9 +138,8 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
|||||||
[StructLayout(LayoutKind.Auto)]
|
[StructLayout(LayoutKind.Auto)]
|
||||||
public struct AsyncUniTaskMethodBuilder<T>
|
public struct AsyncUniTaskMethodBuilder<T>
|
||||||
{
|
{
|
||||||
// cache items.
|
IStateMachineRunnerPromise<T> runnerPromise;
|
||||||
AutoResetUniTaskCompletionSource<T> promise;
|
Exception ex;
|
||||||
IMoveNextRunner runner;
|
|
||||||
T result;
|
T result;
|
||||||
|
|
||||||
// 1. Static Create method.
|
// 1. Static Create method.
|
||||||
@@ -170,106 +151,91 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 2. TaskLike Task property.
|
// 2. TaskLike Task property.
|
||||||
[DebuggerHidden]
|
|
||||||
public UniTask<T> Task
|
public UniTask<T> Task
|
||||||
{
|
{
|
||||||
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
get
|
get
|
||||||
{
|
{
|
||||||
if (promise != null)
|
if (runnerPromise != null)
|
||||||
{
|
{
|
||||||
return promise.Task;
|
return runnerPromise.Task;
|
||||||
}
|
}
|
||||||
|
else if (ex != null)
|
||||||
if (runner == null)
|
{
|
||||||
|
return UniTask.FromException<T>(ex);
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
return UniTask.FromResult(result);
|
return UniTask.FromResult(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
promise = AutoResetUniTaskCompletionSource<T>.Create();
|
|
||||||
return promise.Task;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. SetException
|
// 3. SetException
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void SetException(Exception exception)
|
public void SetException(Exception exception)
|
||||||
{
|
{
|
||||||
// runner is finished, return first.
|
if (runnerPromise == null)
|
||||||
if (runner != null)
|
|
||||||
{
|
{
|
||||||
runner.Return();
|
ex = exception;
|
||||||
runner = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (promise == null)
|
|
||||||
{
|
|
||||||
promise = AutoResetUniTaskCompletionSource<T>.CreateFromException(exception, out _);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
promise.TrySetException(exception);
|
runnerPromise.SetException(exception);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 4. SetResult
|
// 4. SetResult
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void SetResult(T result)
|
public void SetResult(T result)
|
||||||
{
|
{
|
||||||
// runner is finished, return first.
|
if (runnerPromise == null)
|
||||||
if (runner != null)
|
|
||||||
{
|
|
||||||
runner.Return();
|
|
||||||
runner = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (promise == null)
|
|
||||||
{
|
{
|
||||||
this.result = result;
|
this.result = result;
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
else
|
||||||
promise.TrySetResult(result);
|
{
|
||||||
|
runnerPromise.SetResult(result);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 5. AwaitOnCompleted
|
// 5. AwaitOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : INotifyCompletion
|
where TAwaiter : INotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (promise == null)
|
if (runnerPromise == null)
|
||||||
{
|
{
|
||||||
promise = AutoResetUniTaskCompletionSource<T>.Create();
|
AsyncUniTask<TStateMachine, T>.SetStateMachine(ref stateMachine, ref runnerPromise);
|
||||||
}
|
|
||||||
if (runner == null)
|
|
||||||
{
|
|
||||||
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.OnCompleted(runner.CallMoveNext);
|
awaiter.OnCompleted(runnerPromise.MoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 6. AwaitUnsafeOnCompleted
|
// 6. AwaitUnsafeOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[SecuritySafeCritical]
|
[SecuritySafeCritical]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : ICriticalNotifyCompletion
|
where TAwaiter : ICriticalNotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (promise == null)
|
if (runnerPromise == null)
|
||||||
{
|
{
|
||||||
promise = AutoResetUniTaskCompletionSource<T>.Create();
|
AsyncUniTask<TStateMachine, T>.SetStateMachine(ref stateMachine, ref runnerPromise);
|
||||||
}
|
|
||||||
if (runner == null)
|
|
||||||
{
|
|
||||||
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.OnCompleted(runner.CallMoveNext);
|
awaiter.UnsafeOnCompleted(runnerPromise.MoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 7. Start
|
// 7. Start
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void Start<TStateMachine>(ref TStateMachine stateMachine)
|
public void Start<TStateMachine>(ref TStateMachine stateMachine)
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -4,13 +4,15 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.Runtime.CompilerServices;
|
using System.Runtime.CompilerServices;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
using System.Security;
|
using System.Security;
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks.CompilerServices
|
namespace Cysharp.Threading.Tasks.CompilerServices
|
||||||
{
|
{
|
||||||
|
[StructLayout(LayoutKind.Auto)]
|
||||||
public struct AsyncUniTaskVoidMethodBuilder
|
public struct AsyncUniTaskVoidMethodBuilder
|
||||||
{
|
{
|
||||||
IMoveNextRunner runner;
|
IStateMachineRunner runner;
|
||||||
|
|
||||||
// 1. Static Create method.
|
// 1. Static Create method.
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
@@ -33,6 +35,7 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
|||||||
|
|
||||||
// 3. SetException
|
// 3. SetException
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void SetException(Exception exception)
|
public void SetException(Exception exception)
|
||||||
{
|
{
|
||||||
// runner is finished, return first.
|
// runner is finished, return first.
|
||||||
@@ -47,6 +50,7 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
|||||||
|
|
||||||
// 4. SetResult
|
// 4. SetResult
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void SetResult()
|
public void SetResult()
|
||||||
{
|
{
|
||||||
// runner is finished, return.
|
// runner is finished, return.
|
||||||
@@ -59,31 +63,33 @@ namespace Cysharp.Threading.Tasks.CompilerServices
|
|||||||
|
|
||||||
// 5. AwaitOnCompleted
|
// 5. AwaitOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : INotifyCompletion
|
where TAwaiter : INotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (runner == null)
|
if (runner == null)
|
||||||
{
|
{
|
||||||
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
AsyncUniTaskVoid<TStateMachine>.SetStateMachine(ref stateMachine, ref runner);
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.OnCompleted(runner.CallMoveNext);
|
awaiter.OnCompleted(runner.MoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 6. AwaitUnsafeOnCompleted
|
// 6. AwaitUnsafeOnCompleted
|
||||||
[DebuggerHidden]
|
[DebuggerHidden]
|
||||||
[SecuritySafeCritical]
|
[SecuritySafeCritical]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine)
|
||||||
where TAwaiter : ICriticalNotifyCompletion
|
where TAwaiter : ICriticalNotifyCompletion
|
||||||
where TStateMachine : IAsyncStateMachine
|
where TStateMachine : IAsyncStateMachine
|
||||||
{
|
{
|
||||||
if (runner == null)
|
if (runner == null)
|
||||||
{
|
{
|
||||||
runner = MoveNextRunner<TStateMachine>.Create(ref stateMachine);
|
AsyncUniTaskVoid<TStateMachine>.SetStateMachine(ref stateMachine, ref runner);
|
||||||
}
|
}
|
||||||
|
|
||||||
awaiter.OnCompleted(runner.CallMoveNext);
|
awaiter.UnsafeOnCompleted(runner.MoveNext);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 7. Start
|
// 7. Start
|
||||||
|
|||||||
@@ -1,57 +0,0 @@
|
|||||||
|
|
||||||
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Diagnostics;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
using Cysharp.Threading.Tasks.Internal;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks.CompilerServices
|
|
||||||
{
|
|
||||||
internal interface IMoveNextRunner
|
|
||||||
{
|
|
||||||
Action CallMoveNext { get; }
|
|
||||||
void Return();
|
|
||||||
}
|
|
||||||
|
|
||||||
internal sealed class MoveNextRunner<TStateMachine> : IMoveNextRunner, IPromisePoolItem
|
|
||||||
where TStateMachine : IAsyncStateMachine
|
|
||||||
{
|
|
||||||
static PromisePool<MoveNextRunner<TStateMachine>> pool = new PromisePool<MoveNextRunner<TStateMachine>>();
|
|
||||||
|
|
||||||
TStateMachine stateMachine;
|
|
||||||
internal readonly Action callMoveNext;
|
|
||||||
|
|
||||||
public Action CallMoveNext => callMoveNext;
|
|
||||||
|
|
||||||
MoveNextRunner()
|
|
||||||
{
|
|
||||||
callMoveNext = MoveNext;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static MoveNextRunner<TStateMachine> Create(ref TStateMachine stateMachine)
|
|
||||||
{
|
|
||||||
var result = pool.TryRent() ?? new MoveNextRunner<TStateMachine>();
|
|
||||||
result.stateMachine = stateMachine;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
[DebuggerHidden]
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
void MoveNext()
|
|
||||||
{
|
|
||||||
stateMachine.MoveNext();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Return()
|
|
||||||
{
|
|
||||||
pool.TryReturn(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
void IPromisePoolItem.Reset()
|
|
||||||
{
|
|
||||||
stateMachine = default;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -0,0 +1,343 @@
|
|||||||
|
#pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
|
||||||
|
|
||||||
|
using Cysharp.Threading.Tasks.Internal;
|
||||||
|
using System;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Diagnostics;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks.CompilerServices
|
||||||
|
{
|
||||||
|
internal interface IStateMachineRunner
|
||||||
|
{
|
||||||
|
Action MoveNext { get; }
|
||||||
|
void Return();
|
||||||
|
}
|
||||||
|
|
||||||
|
internal interface IStateMachineRunnerPromise : IUniTaskSource
|
||||||
|
{
|
||||||
|
Action MoveNext { get; }
|
||||||
|
UniTask Task { get; }
|
||||||
|
void SetResult();
|
||||||
|
void SetException(Exception exception);
|
||||||
|
}
|
||||||
|
|
||||||
|
internal interface IStateMachineRunnerPromise<T> : IUniTaskSource<T>
|
||||||
|
{
|
||||||
|
Action MoveNext { get; }
|
||||||
|
UniTask<T> Task { get; }
|
||||||
|
void SetResult(T result);
|
||||||
|
void SetException(Exception exception);
|
||||||
|
}
|
||||||
|
|
||||||
|
internal static class StateMachineUtility
|
||||||
|
{
|
||||||
|
public static int GetState(IAsyncStateMachine stateMachine)
|
||||||
|
{
|
||||||
|
var info = stateMachine.GetType().GetFields(System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance)
|
||||||
|
.First(x => x.Name.EndsWith("__state"));
|
||||||
|
return (int)info.GetValue(stateMachine);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal sealed class AsyncUniTaskVoid<TStateMachine> : IStateMachineRunner, ITaskPoolNode<AsyncUniTaskVoid<TStateMachine>>, IUniTaskSource
|
||||||
|
where TStateMachine : IAsyncStateMachine
|
||||||
|
{
|
||||||
|
static TaskPool<AsyncUniTaskVoid<TStateMachine>> pool;
|
||||||
|
|
||||||
|
IAsyncStateMachine stateMachine; // unfortunatelly boxed to fix IL2CPP issue.
|
||||||
|
|
||||||
|
public Action MoveNext { get; }
|
||||||
|
|
||||||
|
public AsyncUniTaskVoid()
|
||||||
|
{
|
||||||
|
MoveNext = Run;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void SetStateMachine(ref TStateMachine stateMachine, ref IStateMachineRunner runnerFieldRef)
|
||||||
|
{
|
||||||
|
if (!pool.TryPop(out var result))
|
||||||
|
{
|
||||||
|
result = new AsyncUniTaskVoid<TStateMachine>();
|
||||||
|
}
|
||||||
|
TaskTracker.TrackActiveTask(result, 3);
|
||||||
|
|
||||||
|
runnerFieldRef = result; // set runner before copied.
|
||||||
|
result.stateMachine = stateMachine; // copy struct StateMachine(in release build).
|
||||||
|
}
|
||||||
|
|
||||||
|
static AsyncUniTaskVoid()
|
||||||
|
{
|
||||||
|
TaskPool.RegisterSizeGetter(typeof(AsyncUniTaskVoid<TStateMachine>), () => pool.Size);
|
||||||
|
}
|
||||||
|
|
||||||
|
public AsyncUniTaskVoid<TStateMachine> NextNode { get; set; }
|
||||||
|
|
||||||
|
public void Return()
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
stateMachine = default;
|
||||||
|
pool.TryPush(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
void Run()
|
||||||
|
{
|
||||||
|
stateMachine.MoveNext();
|
||||||
|
}
|
||||||
|
|
||||||
|
// dummy interface implementation for TaskTracker.
|
||||||
|
|
||||||
|
UniTaskStatus IUniTaskSource.GetStatus(short token)
|
||||||
|
{
|
||||||
|
return UniTaskStatus.Pending;
|
||||||
|
}
|
||||||
|
|
||||||
|
UniTaskStatus IUniTaskSource.UnsafeGetStatus()
|
||||||
|
{
|
||||||
|
return UniTaskStatus.Pending;
|
||||||
|
}
|
||||||
|
|
||||||
|
void IUniTaskSource.OnCompleted(Action<object> continuation, object state, short token)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void IUniTaskSource.GetResult(short token)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal sealed class AsyncUniTask<TStateMachine> : IStateMachineRunnerPromise, IUniTaskSource, ITaskPoolNode<AsyncUniTask<TStateMachine>>
|
||||||
|
where TStateMachine : IAsyncStateMachine
|
||||||
|
{
|
||||||
|
static TaskPool<AsyncUniTask<TStateMachine>> pool;
|
||||||
|
|
||||||
|
IAsyncStateMachine stateMachine; // unfortunatelly boxed to fix IL2CPP issue.
|
||||||
|
|
||||||
|
public Action MoveNext { get; }
|
||||||
|
|
||||||
|
UniTaskCompletionSourceCore<AsyncUnit> core;
|
||||||
|
|
||||||
|
AsyncUniTask()
|
||||||
|
{
|
||||||
|
MoveNext = Run;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void SetStateMachine(ref TStateMachine stateMachine, ref IStateMachineRunnerPromise runnerPromiseFieldRef)
|
||||||
|
{
|
||||||
|
if (!pool.TryPop(out var result))
|
||||||
|
{
|
||||||
|
result = new AsyncUniTask<TStateMachine>();
|
||||||
|
}
|
||||||
|
TaskTracker.TrackActiveTask(result, 3);
|
||||||
|
|
||||||
|
runnerPromiseFieldRef = result; // set runner before copied.
|
||||||
|
result.stateMachine = stateMachine; // copy struct StateMachine(in release build).
|
||||||
|
}
|
||||||
|
|
||||||
|
public AsyncUniTask<TStateMachine> NextNode { get; set; }
|
||||||
|
|
||||||
|
static AsyncUniTask()
|
||||||
|
{
|
||||||
|
TaskPool.RegisterSizeGetter(typeof(AsyncUniTask<TStateMachine>), () => pool.Size);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool TryReturn()
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
core.Reset();
|
||||||
|
stateMachine = default;
|
||||||
|
return pool.TryPush(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
void Run()
|
||||||
|
{
|
||||||
|
stateMachine.MoveNext();
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask Task
|
||||||
|
{
|
||||||
|
[DebuggerHidden]
|
||||||
|
get
|
||||||
|
{
|
||||||
|
return new UniTask(this, core.Version);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public void SetResult()
|
||||||
|
{
|
||||||
|
core.TrySetResult(AsyncUnit.Default);
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public void SetException(Exception exception)
|
||||||
|
{
|
||||||
|
core.TrySetException(exception);
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public void GetResult(short token)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
core.GetResult(token);
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
TryReturn();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public UniTaskStatus GetStatus(short token)
|
||||||
|
{
|
||||||
|
return core.GetStatus(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public UniTaskStatus UnsafeGetStatus()
|
||||||
|
{
|
||||||
|
return core.UnsafeGetStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
||||||
|
{
|
||||||
|
core.OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
~AsyncUniTask()
|
||||||
|
{
|
||||||
|
if (TryReturn())
|
||||||
|
{
|
||||||
|
GC.ReRegisterForFinalize(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal sealed class AsyncUniTask<TStateMachine, T> : IStateMachineRunnerPromise<T>, IUniTaskSource<T>, ITaskPoolNode<AsyncUniTask<TStateMachine, T>>
|
||||||
|
where TStateMachine : IAsyncStateMachine
|
||||||
|
{
|
||||||
|
static TaskPool<AsyncUniTask<TStateMachine, T>> pool;
|
||||||
|
|
||||||
|
IAsyncStateMachine stateMachine; // unfortunatelly boxed to fix IL2CPP issue.
|
||||||
|
|
||||||
|
public Action MoveNext { get; }
|
||||||
|
|
||||||
|
UniTaskCompletionSourceCore<T> core;
|
||||||
|
|
||||||
|
AsyncUniTask()
|
||||||
|
{
|
||||||
|
MoveNext = Run;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void SetStateMachine(ref TStateMachine stateMachine, ref IStateMachineRunnerPromise<T> runnerPromiseFieldRef)
|
||||||
|
{
|
||||||
|
if (!pool.TryPop(out var result))
|
||||||
|
{
|
||||||
|
result = new AsyncUniTask<TStateMachine, T>();
|
||||||
|
}
|
||||||
|
TaskTracker.TrackActiveTask(result, 3);
|
||||||
|
|
||||||
|
runnerPromiseFieldRef = result; // set runner before copied.
|
||||||
|
result.stateMachine = stateMachine; // copy struct StateMachine(in release build).
|
||||||
|
|
||||||
|
// UnityEngine.Debug.Log($"SetStateMachine State:" + StateMachineUtility.GetState(stateMachine));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public AsyncUniTask<TStateMachine, T> NextNode { get; set; }
|
||||||
|
|
||||||
|
static AsyncUniTask()
|
||||||
|
{
|
||||||
|
TaskPool.RegisterSizeGetter(typeof(AsyncUniTask<TStateMachine, T>), () => pool.Size);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool TryReturn()
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
core.Reset();
|
||||||
|
stateMachine = default;
|
||||||
|
return pool.TryPush(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
void Run()
|
||||||
|
{
|
||||||
|
// UnityEngine.Debug.Log($"MoveNext State:" + StateMachineUtility.GetState(stateMachine));
|
||||||
|
stateMachine.MoveNext();
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask<T> Task
|
||||||
|
{
|
||||||
|
[DebuggerHidden]
|
||||||
|
get
|
||||||
|
{
|
||||||
|
return new UniTask<T>(this, core.Version);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public void SetResult(T result)
|
||||||
|
{
|
||||||
|
core.TrySetResult(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public void SetException(Exception exception)
|
||||||
|
{
|
||||||
|
core.TrySetException(exception);
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public T GetResult(short token)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
return core.GetResult(token);
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
TryReturn();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
void IUniTaskSource.GetResult(short token)
|
||||||
|
{
|
||||||
|
GetResult(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public UniTaskStatus GetStatus(short token)
|
||||||
|
{
|
||||||
|
return core.GetStatus(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public UniTaskStatus UnsafeGetStatus()
|
||||||
|
{
|
||||||
|
return core.UnsafeGetStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
[DebuggerHidden]
|
||||||
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
||||||
|
{
|
||||||
|
core.OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
~AsyncUniTask()
|
||||||
|
{
|
||||||
|
if (TryReturn())
|
||||||
|
{
|
||||||
|
GC.ReRegisterForFinalize(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@@ -12,24 +12,35 @@ namespace Cysharp.Threading.Tasks
|
|||||||
{
|
{
|
||||||
public static class EnumeratorAsyncExtensions
|
public static class EnumeratorAsyncExtensions
|
||||||
{
|
{
|
||||||
public static UniTask.Awaiter GetAwaiter(this IEnumerator enumerator)
|
public static UniTask.Awaiter GetAwaiter<T>(this T enumerator)
|
||||||
|
where T : IEnumerator
|
||||||
{
|
{
|
||||||
return new UniTask(EnumeratorPromise.Create(enumerator, PlayerLoopTiming.Update, CancellationToken.None, out var token), token).GetAwaiter();
|
var e = (IEnumerator)enumerator;
|
||||||
|
Error.ThrowArgumentNullException(e, nameof(enumerator));
|
||||||
|
return new UniTask(EnumeratorPromise.Create(e, PlayerLoopTiming.Update, CancellationToken.None, out var token), token).GetAwaiter();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask ToUniTask(this IEnumerator enumerator)
|
public static UniTask WithCancellation(this IEnumerator enumerator, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
return new UniTask(EnumeratorPromise.Create(enumerator, PlayerLoopTiming.Update, CancellationToken.None, out var token), token);
|
Error.ThrowArgumentNullException(enumerator, nameof(enumerator));
|
||||||
|
return new UniTask(EnumeratorPromise.Create(enumerator, PlayerLoopTiming.Update, cancellationToken, out var token), token);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask ConfigureAwait(this IEnumerator enumerator, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellationToken = default(CancellationToken))
|
public static UniTask ToUniTask(this IEnumerator enumerator, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellationToken = default(CancellationToken))
|
||||||
{
|
{
|
||||||
|
Error.ThrowArgumentNullException(enumerator, nameof(enumerator));
|
||||||
return new UniTask(EnumeratorPromise.Create(enumerator, timing, cancellationToken, out var token), token);
|
return new UniTask(EnumeratorPromise.Create(enumerator, timing, cancellationToken, out var token), token);
|
||||||
}
|
}
|
||||||
|
|
||||||
class EnumeratorPromise : IUniTaskSource, IPlayerLoopItem, IPromisePoolItem
|
sealed class EnumeratorPromise : IUniTaskSource, IPlayerLoopItem, ITaskPoolNode<EnumeratorPromise>
|
||||||
{
|
{
|
||||||
static readonly PromisePool<EnumeratorPromise> pool = new PromisePool<EnumeratorPromise>();
|
static TaskPool<EnumeratorPromise> pool;
|
||||||
|
public EnumeratorPromise NextNode { get; set; }
|
||||||
|
|
||||||
|
static EnumeratorPromise()
|
||||||
|
{
|
||||||
|
TaskPool.RegisterSizeGetter(typeof(EnumeratorPromise), () => pool.Size);
|
||||||
|
}
|
||||||
|
|
||||||
IEnumerator innerEnumerator;
|
IEnumerator innerEnumerator;
|
||||||
CancellationToken cancellationToken;
|
CancellationToken cancellationToken;
|
||||||
@@ -47,13 +58,15 @@ namespace Cysharp.Threading.Tasks
|
|||||||
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
||||||
}
|
}
|
||||||
|
|
||||||
var result = pool.TryRent() ?? new EnumeratorPromise();
|
if (!pool.TryPop(out var result))
|
||||||
|
{
|
||||||
|
result = new EnumeratorPromise();
|
||||||
|
}
|
||||||
|
TaskTracker.TrackActiveTask(result, 3);
|
||||||
|
|
||||||
result.innerEnumerator = ConsumeEnumerator(innerEnumerator);
|
result.innerEnumerator = ConsumeEnumerator(innerEnumerator);
|
||||||
result.cancellationToken = cancellationToken;
|
result.cancellationToken = cancellationToken;
|
||||||
|
|
||||||
TaskTracker.TrackActiveTask(result, 3);
|
|
||||||
|
|
||||||
PlayerLoopHelper.AddAction(timing, result);
|
PlayerLoopHelper.AddAction(timing, result);
|
||||||
|
|
||||||
token = result.core.Version;
|
token = result.core.Version;
|
||||||
@@ -64,12 +77,11 @@ namespace Cysharp.Threading.Tasks
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.GetResult(token);
|
core.GetResult(token);
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
pool.TryReturn(this);
|
TryReturn();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -113,16 +125,18 @@ namespace Cysharp.Threading.Tasks
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Reset()
|
bool TryReturn()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
core.Reset();
|
core.Reset();
|
||||||
innerEnumerator = default;
|
innerEnumerator = default;
|
||||||
cancellationToken = default;
|
cancellationToken = default;
|
||||||
|
return pool.TryPush(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
~EnumeratorPromise()
|
~EnumeratorPromise()
|
||||||
{
|
{
|
||||||
if (pool.TryReturn(this))
|
if (TryReturn())
|
||||||
{
|
{
|
||||||
GC.ReRegisterForFinalize(this);
|
GC.ReRegisterForFinalize(this);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,14 +20,16 @@ namespace Cysharp.Threading.Tasks
|
|||||||
return new AsyncOperationHandleAwaiter(handle);
|
return new AsyncOperationHandleAwaiter(handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask ToUniTask(this AsyncOperationHandle handle)
|
public static UniTask WithCancellation(this AsyncOperationHandle handle, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
return new UniTask(AsyncOperationHandleConfiguredSource.Create(handle, PlayerLoopTiming.Update, null, CancellationToken.None, out var token), token);
|
if (handle.IsDone) return UniTask.CompletedTask;
|
||||||
|
return new UniTask(AsyncOperationHandleWithCancellationSource.Create(handle, cancellationToken, out var token), token);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask ConfigureAwait(this AsyncOperationHandle handle, IProgress<float> progress = null, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellation = default(CancellationToken))
|
public static UniTask ToUniTask(this AsyncOperationHandle handle, IProgress<float> progress = null, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellationToken = default(CancellationToken))
|
||||||
{
|
{
|
||||||
return new UniTask(AsyncOperationHandleConfiguredSource.Create(handle, timing, progress, cancellation, out var token), token);
|
if (handle.IsDone) return UniTask.CompletedTask;
|
||||||
|
return new UniTask(AsyncOperationHandleConfiguredSource.Create(handle, timing, progress, cancellationToken, out var token), token);
|
||||||
}
|
}
|
||||||
|
|
||||||
public struct AsyncOperationHandleAwaiter : ICriticalNotifyCompletion
|
public struct AsyncOperationHandleAwaiter : ICriticalNotifyCompletion
|
||||||
@@ -70,14 +72,146 @@ namespace Cysharp.Threading.Tasks
|
|||||||
public void UnsafeOnCompleted(Action continuation)
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
{
|
{
|
||||||
Error.ThrowWhenContinuationIsAlreadyRegistered(continuationAction);
|
Error.ThrowWhenContinuationIsAlreadyRegistered(continuationAction);
|
||||||
continuationAction = continuation.AsFuncOfT<AsyncOperationHandle>(); // allocate delegate.
|
continuationAction = PooledDelegate<AsyncOperationHandle>.Create(continuation);
|
||||||
handle.Completed += continuationAction;
|
handle.Completed += continuationAction;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class AsyncOperationHandleConfiguredSource : IUniTaskSource, IPlayerLoopItem, IPromisePoolItem
|
sealed class AsyncOperationHandleWithCancellationSource : IUniTaskSource, IPlayerLoopItem, ITaskPoolNode<AsyncOperationHandleWithCancellationSource>
|
||||||
{
|
{
|
||||||
static readonly PromisePool<AsyncOperationHandleConfiguredSource> pool = new PromisePool<AsyncOperationHandleConfiguredSource>();
|
static TaskPool<AsyncOperationHandleWithCancellationSource> pool;
|
||||||
|
public AsyncOperationHandleWithCancellationSource NextNode { get; set; }
|
||||||
|
|
||||||
|
static AsyncOperationHandleWithCancellationSource()
|
||||||
|
{
|
||||||
|
TaskPool.RegisterSizeGetter(typeof(AsyncOperationHandleWithCancellationSource), () => pool.Size);
|
||||||
|
}
|
||||||
|
|
||||||
|
readonly Action<AsyncOperationHandle> continuationAction;
|
||||||
|
AsyncOperationHandle handle;
|
||||||
|
CancellationToken cancellationToken;
|
||||||
|
bool completed;
|
||||||
|
|
||||||
|
UniTaskCompletionSourceCore<AsyncUnit> core;
|
||||||
|
|
||||||
|
AsyncOperationHandleWithCancellationSource()
|
||||||
|
{
|
||||||
|
continuationAction = Continuation;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static IUniTaskSource Create(AsyncOperationHandle handle, CancellationToken cancellationToken, out short token)
|
||||||
|
{
|
||||||
|
if (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!pool.TryPop(out var result))
|
||||||
|
{
|
||||||
|
result = new AsyncOperationHandleWithCancellationSource();
|
||||||
|
}
|
||||||
|
|
||||||
|
result.handle = handle;
|
||||||
|
result.cancellationToken = cancellationToken;
|
||||||
|
result.completed = false;
|
||||||
|
|
||||||
|
TaskTracker.TrackActiveTask(result, 3);
|
||||||
|
|
||||||
|
PlayerLoopHelper.AddAction(PlayerLoopTiming.Update, result);
|
||||||
|
|
||||||
|
handle.Completed += result.continuationAction;
|
||||||
|
|
||||||
|
token = result.core.Version;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Continuation(AsyncOperationHandle _)
|
||||||
|
{
|
||||||
|
handle.Completed -= continuationAction;
|
||||||
|
|
||||||
|
if (completed)
|
||||||
|
{
|
||||||
|
TryReturn();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
completed = true;
|
||||||
|
if (handle.Status == AsyncOperationStatus.Failed)
|
||||||
|
{
|
||||||
|
core.TrySetException(handle.OperationException);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
core.TrySetResult(AsyncUnit.Default);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void GetResult(short token)
|
||||||
|
{
|
||||||
|
core.GetResult(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTaskStatus GetStatus(short token)
|
||||||
|
{
|
||||||
|
return core.GetStatus(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTaskStatus UnsafeGetStatus()
|
||||||
|
{
|
||||||
|
return core.UnsafeGetStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
||||||
|
{
|
||||||
|
core.OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool MoveNext()
|
||||||
|
{
|
||||||
|
if (completed)
|
||||||
|
{
|
||||||
|
TryReturn();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
completed = true;
|
||||||
|
core.TrySetCanceled(cancellationToken);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool TryReturn()
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
core.Reset();
|
||||||
|
handle = default;
|
||||||
|
cancellationToken = default;
|
||||||
|
return pool.TryPush(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
~AsyncOperationHandleWithCancellationSource()
|
||||||
|
{
|
||||||
|
if (TryReturn())
|
||||||
|
{
|
||||||
|
GC.ReRegisterForFinalize(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class AsyncOperationHandleConfiguredSource : IUniTaskSource, IPlayerLoopItem, ITaskPoolNode<AsyncOperationHandleConfiguredSource>
|
||||||
|
{
|
||||||
|
static TaskPool<AsyncOperationHandleConfiguredSource> pool;
|
||||||
|
public AsyncOperationHandleConfiguredSource NextNode { get; set; }
|
||||||
|
|
||||||
|
static AsyncOperationHandleConfiguredSource()
|
||||||
|
{
|
||||||
|
TaskPool.RegisterSizeGetter(typeof(AsyncOperationHandleConfiguredSource), () => pool.Size);
|
||||||
|
}
|
||||||
|
|
||||||
AsyncOperationHandle handle;
|
AsyncOperationHandle handle;
|
||||||
IProgress<float> progress;
|
IProgress<float> progress;
|
||||||
@@ -97,7 +231,10 @@ namespace Cysharp.Threading.Tasks
|
|||||||
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
||||||
}
|
}
|
||||||
|
|
||||||
var result = pool.TryRent() ?? new AsyncOperationHandleConfiguredSource();
|
if (!pool.TryPop(out var result))
|
||||||
|
{
|
||||||
|
result = new AsyncOperationHandleConfiguredSource();
|
||||||
|
}
|
||||||
|
|
||||||
result.handle = handle;
|
result.handle = handle;
|
||||||
result.progress = progress;
|
result.progress = progress;
|
||||||
@@ -115,12 +252,12 @@ namespace Cysharp.Threading.Tasks
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
core.GetResult(token);
|
core.GetResult(token);
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
pool.TryReturn(this);
|
TryReturn();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -168,17 +305,19 @@ namespace Cysharp.Threading.Tasks
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Reset()
|
bool TryReturn()
|
||||||
{
|
{
|
||||||
core.Reset();
|
core.Reset();
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
handle = default;
|
handle = default;
|
||||||
progress = default;
|
progress = default;
|
||||||
cancellationToken = default;
|
cancellationToken = default;
|
||||||
|
return pool.TryPush(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
~AsyncOperationHandleConfiguredSource()
|
~AsyncOperationHandleConfiguredSource()
|
||||||
{
|
{
|
||||||
if (pool.TryReturn(this))
|
if (TryReturn())
|
||||||
{
|
{
|
||||||
GC.ReRegisterForFinalize(this);
|
GC.ReRegisterForFinalize(this);
|
||||||
}
|
}
|
||||||
@@ -194,14 +333,16 @@ namespace Cysharp.Threading.Tasks
|
|||||||
return new AsyncOperationHandleAwaiter<T>(handle);
|
return new AsyncOperationHandleAwaiter<T>(handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<T> ToUniTask<T>(this AsyncOperationHandle<T> handle)
|
public static UniTask<T> WithCancellation<T>(this AsyncOperationHandle<T> handle, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
return new UniTask<T>(AsyncOperationHandleConfiguredSource<T>.Create(handle, PlayerLoopTiming.Update, null, CancellationToken.None, out var token), token);
|
if (handle.IsDone) return UniTask.FromResult(handle.Result);
|
||||||
|
return new UniTask<T>(AsyncOperationHandleWithCancellationSource<T>.Create(handle, cancellationToken, out var token), token);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<T> ConfigureAwait<T>(this AsyncOperationHandle<T> handle, IProgress<float> progress = null, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellation = default(CancellationToken))
|
public static UniTask<T> ToUniTask<T>(this AsyncOperationHandle<T> handle, IProgress<float> progress = null, PlayerLoopTiming timing = PlayerLoopTiming.Update, CancellationToken cancellationToken = default(CancellationToken))
|
||||||
{
|
{
|
||||||
return new UniTask<T>(AsyncOperationHandleConfiguredSource<T>.Create(handle, timing, progress, cancellation, out var token), token);
|
if (handle.IsDone) return UniTask.FromResult(handle.Result);
|
||||||
|
return new UniTask<T>(AsyncOperationHandleConfiguredSource<T>.Create(handle, timing, progress, cancellationToken, out var token), token);
|
||||||
}
|
}
|
||||||
|
|
||||||
public struct AsyncOperationHandleAwaiter<T> : ICriticalNotifyCompletion
|
public struct AsyncOperationHandleAwaiter<T> : ICriticalNotifyCompletion
|
||||||
@@ -245,14 +386,151 @@ namespace Cysharp.Threading.Tasks
|
|||||||
public void UnsafeOnCompleted(Action continuation)
|
public void UnsafeOnCompleted(Action continuation)
|
||||||
{
|
{
|
||||||
Error.ThrowWhenContinuationIsAlreadyRegistered(continuationAction);
|
Error.ThrowWhenContinuationIsAlreadyRegistered(continuationAction);
|
||||||
continuationAction = continuation.AsFuncOfT<AsyncOperationHandle>(); // allocate delegate.
|
continuationAction = PooledDelegate<AsyncOperationHandle>.Create(continuation);
|
||||||
handle.CompletedTypeless += continuationAction;
|
handle.CompletedTypeless += continuationAction;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class AsyncOperationHandleConfiguredSource<T> : IUniTaskSource<T>, IPlayerLoopItem, IPromisePoolItem
|
sealed class AsyncOperationHandleWithCancellationSource<T> : IUniTaskSource<T>, IPlayerLoopItem, ITaskPoolNode<AsyncOperationHandleWithCancellationSource<T>>
|
||||||
{
|
{
|
||||||
static readonly PromisePool<AsyncOperationHandleConfiguredSource<T>> pool = new PromisePool<AsyncOperationHandleConfiguredSource<T>>();
|
static TaskPool<AsyncOperationHandleWithCancellationSource<T>> pool;
|
||||||
|
public AsyncOperationHandleWithCancellationSource<T> NextNode { get; set; }
|
||||||
|
|
||||||
|
static AsyncOperationHandleWithCancellationSource()
|
||||||
|
{
|
||||||
|
TaskPool.RegisterSizeGetter(typeof(AsyncOperationHandleWithCancellationSource<T>), () => pool.Size);
|
||||||
|
}
|
||||||
|
|
||||||
|
readonly Action<AsyncOperationHandle<T>> continuationAction;
|
||||||
|
AsyncOperationHandle<T> handle;
|
||||||
|
CancellationToken cancellationToken;
|
||||||
|
bool completed;
|
||||||
|
|
||||||
|
UniTaskCompletionSourceCore<T> core;
|
||||||
|
|
||||||
|
AsyncOperationHandleWithCancellationSource()
|
||||||
|
{
|
||||||
|
continuationAction = Continuation;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static IUniTaskSource<T> Create(AsyncOperationHandle<T> handle, CancellationToken cancellationToken, out short token)
|
||||||
|
{
|
||||||
|
if (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
return AutoResetUniTaskCompletionSource<T>.CreateFromCanceled(cancellationToken, out token);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!pool.TryPop(out var result))
|
||||||
|
{
|
||||||
|
result = new AsyncOperationHandleWithCancellationSource<T>();
|
||||||
|
}
|
||||||
|
|
||||||
|
result.handle = handle;
|
||||||
|
result.cancellationToken = cancellationToken;
|
||||||
|
result.completed = false;
|
||||||
|
|
||||||
|
TaskTracker.TrackActiveTask(result, 3);
|
||||||
|
|
||||||
|
PlayerLoopHelper.AddAction(PlayerLoopTiming.Update, result);
|
||||||
|
|
||||||
|
handle.Completed += result.continuationAction;
|
||||||
|
|
||||||
|
token = result.core.Version;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Continuation(AsyncOperationHandle<T> _)
|
||||||
|
{
|
||||||
|
handle.Completed -= continuationAction;
|
||||||
|
|
||||||
|
if (completed)
|
||||||
|
{
|
||||||
|
TryReturn();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
completed = true;
|
||||||
|
if (handle.Status == AsyncOperationStatus.Failed)
|
||||||
|
{
|
||||||
|
core.TrySetException(handle.OperationException);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
core.TrySetResult(handle.Result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public T GetResult(short token)
|
||||||
|
{
|
||||||
|
return core.GetResult(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
void IUniTaskSource.GetResult(short token)
|
||||||
|
{
|
||||||
|
GetResult(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTaskStatus GetStatus(short token)
|
||||||
|
{
|
||||||
|
return core.GetStatus(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTaskStatus UnsafeGetStatus()
|
||||||
|
{
|
||||||
|
return core.UnsafeGetStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
||||||
|
{
|
||||||
|
core.OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool MoveNext()
|
||||||
|
{
|
||||||
|
if (completed)
|
||||||
|
{
|
||||||
|
TryReturn();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
completed = true;
|
||||||
|
core.TrySetCanceled(cancellationToken);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool TryReturn()
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
core.Reset();
|
||||||
|
handle = default;
|
||||||
|
cancellationToken = default;
|
||||||
|
return pool.TryPush(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
~AsyncOperationHandleWithCancellationSource()
|
||||||
|
{
|
||||||
|
if (TryReturn())
|
||||||
|
{
|
||||||
|
GC.ReRegisterForFinalize(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class AsyncOperationHandleConfiguredSource<T> : IUniTaskSource<T>, IPlayerLoopItem, ITaskPoolNode<AsyncOperationHandleConfiguredSource<T>>
|
||||||
|
{
|
||||||
|
static TaskPool<AsyncOperationHandleConfiguredSource<T>> pool;
|
||||||
|
public AsyncOperationHandleConfiguredSource<T> NextNode { get; set; }
|
||||||
|
|
||||||
|
static AsyncOperationHandleConfiguredSource()
|
||||||
|
{
|
||||||
|
TaskPool.RegisterSizeGetter(typeof(AsyncOperationHandleConfiguredSource<T>), () => pool.Size);
|
||||||
|
}
|
||||||
|
|
||||||
AsyncOperationHandle<T> handle;
|
AsyncOperationHandle<T> handle;
|
||||||
IProgress<float> progress;
|
IProgress<float> progress;
|
||||||
@@ -272,7 +550,10 @@ namespace Cysharp.Threading.Tasks
|
|||||||
return AutoResetUniTaskCompletionSource<T>.CreateFromCanceled(cancellationToken, out token);
|
return AutoResetUniTaskCompletionSource<T>.CreateFromCanceled(cancellationToken, out token);
|
||||||
}
|
}
|
||||||
|
|
||||||
var result = pool.TryRent() ?? new AsyncOperationHandleConfiguredSource<T>();
|
if (!pool.TryPop(out var result))
|
||||||
|
{
|
||||||
|
result = new AsyncOperationHandleConfiguredSource<T>();
|
||||||
|
}
|
||||||
|
|
||||||
result.handle = handle;
|
result.handle = handle;
|
||||||
result.progress = progress;
|
result.progress = progress;
|
||||||
@@ -290,13 +571,11 @@ namespace Cysharp.Threading.Tasks
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
TaskTracker.RemoveTracking(this);
|
|
||||||
|
|
||||||
return core.GetResult(token);
|
return core.GetResult(token);
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
pool.TryReturn(this);
|
TryReturn();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -349,17 +628,19 @@ namespace Cysharp.Threading.Tasks
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Reset()
|
bool TryReturn()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
core.Reset();
|
core.Reset();
|
||||||
handle = default;
|
handle = default;
|
||||||
progress = default;
|
progress = default;
|
||||||
cancellationToken = default;
|
cancellationToken = default;
|
||||||
|
return pool.TryPush(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
~AsyncOperationHandleConfiguredSource()
|
~AsyncOperationHandleConfiguredSource()
|
||||||
{
|
{
|
||||||
if (pool.TryReturn(this))
|
if (TryReturn())
|
||||||
{
|
{
|
||||||
GC.ReRegisterForFinalize(this);
|
GC.ReRegisterForFinalize(this);
|
||||||
}
|
}
|
||||||
|
|||||||
332
src/UniTask/Assets/Plugins/UniTask/Runtime/External/DoTweenAsyncExtensions.cs
vendored
Normal file
332
src/UniTask/Assets/Plugins/UniTask/Runtime/External/DoTweenAsyncExtensions.cs
vendored
Normal file
@@ -0,0 +1,332 @@
|
|||||||
|
// asmdef Version Defines, enabled when com.demigiant.dotween is imported.
|
||||||
|
|
||||||
|
#if UNITASK_DOTWEEN_SUPPORT
|
||||||
|
|
||||||
|
using Cysharp.Threading.Tasks.Internal;
|
||||||
|
using DG.Tweening;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Concurrent;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks
|
||||||
|
{
|
||||||
|
// The idea of TweenCancelBehaviour is borrowed from https://www.shibuya24.info/entry/dotween_async_await
|
||||||
|
public enum TweenCancelBehaviour
|
||||||
|
{
|
||||||
|
Kill,
|
||||||
|
KillWithCompleteCallback,
|
||||||
|
Complete,
|
||||||
|
CompleteWithSeqeunceCallback,
|
||||||
|
CancelAwait,
|
||||||
|
|
||||||
|
// AndCancelAwait
|
||||||
|
KillAndCancelAwait,
|
||||||
|
KillWithCompleteCallbackAndCancelAwait,
|
||||||
|
CompleteAndCancelAwait,
|
||||||
|
CompleteWithSeqeunceCallbackAndCancelAwait
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class DOTweenAsyncExtensions
|
||||||
|
{
|
||||||
|
public static TweenAwaiter GetAwaiter(this Tween tween)
|
||||||
|
{
|
||||||
|
return new TweenAwaiter(tween);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static UniTask WithCancellation(this Tween tween, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
Error.ThrowArgumentNullException(tween, nameof(tween));
|
||||||
|
|
||||||
|
if (!tween.IsActive()) return UniTask.CompletedTask;
|
||||||
|
return new UniTask(TweenConfiguredSource.Create(tween, TweenCancelBehaviour.Kill, cancellationToken, out var token), token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static UniTask ToUniTask(this Tween tween, TweenCancelBehaviour tweenCancelBehaviour = TweenCancelBehaviour.Kill, CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
Error.ThrowArgumentNullException(tween, nameof(tween));
|
||||||
|
|
||||||
|
if (!tween.IsActive()) return UniTask.CompletedTask;
|
||||||
|
return new UniTask(TweenConfiguredSource.Create(tween, tweenCancelBehaviour, cancellationToken, out var token), token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct TweenAwaiter : ICriticalNotifyCompletion
|
||||||
|
{
|
||||||
|
readonly Tween tween;
|
||||||
|
|
||||||
|
// killed(non active) as completed.
|
||||||
|
public bool IsCompleted => !tween.IsActive();
|
||||||
|
|
||||||
|
public TweenAwaiter(Tween tween)
|
||||||
|
{
|
||||||
|
this.tween = tween;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TweenAwaiter GetAwaiter()
|
||||||
|
{
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void GetResult()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(System.Action continuation)
|
||||||
|
{
|
||||||
|
UnsafeOnCompleted(continuation);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnsafeOnCompleted(System.Action continuation)
|
||||||
|
{
|
||||||
|
// onKill is called after OnCompleted, both Complete(false/true) and Kill(false/true).
|
||||||
|
tween.onKill = PooledTweenCallback.Create(continuation);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class TweenConfiguredSource : IUniTaskSource, ITaskPoolNode<TweenConfiguredSource>
|
||||||
|
{
|
||||||
|
static TaskPool<TweenConfiguredSource> pool;
|
||||||
|
public TweenConfiguredSource NextNode { get; set; }
|
||||||
|
|
||||||
|
static TweenConfiguredSource()
|
||||||
|
{
|
||||||
|
TaskPool.RegisterSizeGetter(typeof(TweenConfiguredSource), () => pool.Size);
|
||||||
|
}
|
||||||
|
|
||||||
|
static readonly TweenCallback EmptyTweenCallback = () => { };
|
||||||
|
|
||||||
|
readonly TweenCallback onKillDelegate;
|
||||||
|
readonly TweenCallback onUpdateDelegate;
|
||||||
|
|
||||||
|
Tween tween;
|
||||||
|
TweenCancelBehaviour cancelBehaviour;
|
||||||
|
CancellationToken cancellationToken;
|
||||||
|
bool canceled;
|
||||||
|
|
||||||
|
TweenCallback originalUpdateAction;
|
||||||
|
UniTaskCompletionSourceCore<AsyncUnit> core;
|
||||||
|
|
||||||
|
TweenConfiguredSource()
|
||||||
|
{
|
||||||
|
onKillDelegate = OnKill;
|
||||||
|
onUpdateDelegate = OnUpdate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static IUniTaskSource Create(Tween tween, TweenCancelBehaviour cancelBehaviour, CancellationToken cancellationToken, out short token)
|
||||||
|
{
|
||||||
|
if (cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
DoCancelBeforeCreate(tween, cancelBehaviour);
|
||||||
|
return AutoResetUniTaskCompletionSource.CreateFromCanceled(cancellationToken, out token);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!pool.TryPop(out var result))
|
||||||
|
{
|
||||||
|
result = new TweenConfiguredSource();
|
||||||
|
}
|
||||||
|
|
||||||
|
result.tween = tween;
|
||||||
|
result.cancelBehaviour = cancelBehaviour;
|
||||||
|
result.cancellationToken = cancellationToken;
|
||||||
|
|
||||||
|
result.originalUpdateAction = tween.onUpdate;
|
||||||
|
result.canceled = false;
|
||||||
|
|
||||||
|
if (result.originalUpdateAction == result.onUpdateDelegate)
|
||||||
|
{
|
||||||
|
result.originalUpdateAction = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
tween.onUpdate = result.onUpdateDelegate;
|
||||||
|
tween.onKill = result.onKillDelegate;
|
||||||
|
|
||||||
|
TaskTracker.TrackActiveTask(result, 3);
|
||||||
|
|
||||||
|
token = result.core.Version;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void OnKill()
|
||||||
|
{
|
||||||
|
if (canceled)
|
||||||
|
{
|
||||||
|
core.TrySetCanceled(cancellationToken);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
core.TrySetResult(AsyncUnit.Default);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void OnUpdate()
|
||||||
|
{
|
||||||
|
originalUpdateAction?.Invoke();
|
||||||
|
|
||||||
|
if (!cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (this.cancelBehaviour)
|
||||||
|
{
|
||||||
|
case TweenCancelBehaviour.Kill:
|
||||||
|
default:
|
||||||
|
this.tween.Kill(false);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.KillAndCancelAwait:
|
||||||
|
this.canceled = true;
|
||||||
|
this.tween.Kill(false);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.KillWithCompleteCallback:
|
||||||
|
this.tween.Kill(true);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.KillWithCompleteCallbackAndCancelAwait:
|
||||||
|
this.canceled = true;
|
||||||
|
this.tween.Kill(true);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.Complete:
|
||||||
|
this.tween.Complete(false);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.CompleteAndCancelAwait:
|
||||||
|
this.canceled = true;
|
||||||
|
this.tween.Complete(false);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.CompleteWithSeqeunceCallback:
|
||||||
|
this.tween.Complete(true);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.CompleteWithSeqeunceCallbackAndCancelAwait:
|
||||||
|
this.canceled = true;
|
||||||
|
this.tween.Complete(true);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.CancelAwait:
|
||||||
|
this.tween.onKill = EmptyTweenCallback; // replace to empty(avoid callback after Canceled(instance is returned to pool.)
|
||||||
|
this.core.TrySetCanceled(this.cancellationToken);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void DoCancelBeforeCreate(Tween tween, TweenCancelBehaviour tweenCancelBehaviour)
|
||||||
|
{
|
||||||
|
|
||||||
|
switch (tweenCancelBehaviour)
|
||||||
|
{
|
||||||
|
case TweenCancelBehaviour.Kill:
|
||||||
|
default:
|
||||||
|
tween.Kill(false);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.KillAndCancelAwait:
|
||||||
|
tween.Kill(false);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.KillWithCompleteCallback:
|
||||||
|
tween.Kill(true);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.KillWithCompleteCallbackAndCancelAwait:
|
||||||
|
tween.Kill(true);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.Complete:
|
||||||
|
tween.Complete(false);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.CompleteAndCancelAwait:
|
||||||
|
tween.Complete(false);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.CompleteWithSeqeunceCallback:
|
||||||
|
tween.Complete(true);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.CompleteWithSeqeunceCallbackAndCancelAwait:
|
||||||
|
tween.Complete(true);
|
||||||
|
break;
|
||||||
|
case TweenCancelBehaviour.CancelAwait:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void GetResult(short token)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
core.GetResult(token);
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
TryReturn();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTaskStatus GetStatus(short token)
|
||||||
|
{
|
||||||
|
return core.GetStatus(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTaskStatus UnsafeGetStatus()
|
||||||
|
{
|
||||||
|
return core.UnsafeGetStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnCompleted(Action<object> continuation, object state, short token)
|
||||||
|
{
|
||||||
|
core.OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool TryReturn()
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
core.Reset();
|
||||||
|
tween.onUpdate = originalUpdateAction;
|
||||||
|
tween.onKill = null;
|
||||||
|
tween = default;
|
||||||
|
cancellationToken = default;
|
||||||
|
originalUpdateAction = default;
|
||||||
|
return pool.TryPush(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
~TweenConfiguredSource()
|
||||||
|
{
|
||||||
|
if (TryReturn())
|
||||||
|
{
|
||||||
|
GC.ReRegisterForFinalize(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class PooledTweenCallback
|
||||||
|
{
|
||||||
|
static readonly ConcurrentQueue<PooledTweenCallback> pool = new ConcurrentQueue<PooledTweenCallback>();
|
||||||
|
|
||||||
|
readonly TweenCallback runDelegate;
|
||||||
|
|
||||||
|
Action continuation;
|
||||||
|
|
||||||
|
|
||||||
|
PooledTweenCallback()
|
||||||
|
{
|
||||||
|
runDelegate = Run;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public static TweenCallback Create(Action continuation)
|
||||||
|
{
|
||||||
|
if (!pool.TryDequeue(out var item))
|
||||||
|
{
|
||||||
|
item = new PooledTweenCallback();
|
||||||
|
}
|
||||||
|
|
||||||
|
item.continuation = continuation;
|
||||||
|
return item.runDelegate;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
void Run()
|
||||||
|
{
|
||||||
|
var call = continuation;
|
||||||
|
continuation = null;
|
||||||
|
if (call != null)
|
||||||
|
{
|
||||||
|
pool.Enqueue(this);
|
||||||
|
call.Invoke();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
11
src/UniTask/Assets/Plugins/UniTask/Runtime/External/DoTweenAsyncExtensions.cs.meta
vendored
Normal file
11
src/UniTask/Assets/Plugins/UniTask/Runtime/External/DoTweenAsyncExtensions.cs.meta
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
fileFormatVersion: 2
|
||||||
|
guid: 1f448d5bc5b232e4f98d89d5d1832e8e
|
||||||
|
MonoImporter:
|
||||||
|
externalObjects: {}
|
||||||
|
serializedVersion: 2
|
||||||
|
defaultReferences: []
|
||||||
|
executionOrder: 0
|
||||||
|
icon: {instanceID: 0}
|
||||||
|
userData:
|
||||||
|
assetBundleName:
|
||||||
|
assetBundleVariant:
|
||||||
@@ -28,6 +28,12 @@ namespace Cysharp.Threading.Tasks
|
|||||||
IUniTaskOrderedAsyncEnumerable<TElement> CreateOrderedEnumerable<TKey>(Func<TElement, CancellationToken, UniTask<TKey>> keySelector, IComparer<TKey> comparer, bool descending);
|
IUniTaskOrderedAsyncEnumerable<TElement> CreateOrderedEnumerable<TKey>(Func<TElement, CancellationToken, UniTask<TKey>> keySelector, IComparer<TKey> comparer, bool descending);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public interface IConnectableUniTaskAsyncEnumerable<out T> : IUniTaskAsyncEnumerable<T>
|
||||||
|
{
|
||||||
|
IDisposable Connect();
|
||||||
|
}
|
||||||
|
|
||||||
|
// don't use AsyncGrouping.
|
||||||
//public interface IUniTaskAsyncGrouping<out TKey, out TElement> : IUniTaskAsyncEnumerable<TElement>
|
//public interface IUniTaskAsyncGrouping<out TKey, out TElement> : IUniTaskAsyncEnumerable<TElement>
|
||||||
//{
|
//{
|
||||||
// TKey Key { get; }
|
// TKey Key { get; }
|
||||||
|
|||||||
@@ -19,17 +19,75 @@ namespace Cysharp.Threading.Tasks
|
|||||||
|
|
||||||
// similar as IValueTaskSource
|
// similar as IValueTaskSource
|
||||||
public interface IUniTaskSource
|
public interface IUniTaskSource
|
||||||
|
#if !UNITY_2018_3_OR_NEWER && !NETSTANDARD2_0
|
||||||
|
: System.Threading.Tasks.Sources.IValueTaskSource
|
||||||
|
#pragma warning disable CS0108
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
UniTaskStatus GetStatus(short token);
|
UniTaskStatus GetStatus(short token);
|
||||||
void OnCompleted(Action<object> continuation, object state, short token);
|
void OnCompleted(Action<object> continuation, object state, short token);
|
||||||
void GetResult(short token);
|
void GetResult(short token);
|
||||||
|
|
||||||
UniTaskStatus UnsafeGetStatus(); // only for debug use.
|
UniTaskStatus UnsafeGetStatus(); // only for debug use.
|
||||||
|
|
||||||
|
#if !UNITY_2018_3_OR_NEWER && !NETSTANDARD2_0
|
||||||
|
#pragma warning restore CS0108
|
||||||
|
|
||||||
|
System.Threading.Tasks.Sources.ValueTaskSourceStatus System.Threading.Tasks.Sources.IValueTaskSource.GetStatus(short token)
|
||||||
|
{
|
||||||
|
return (System.Threading.Tasks.Sources.ValueTaskSourceStatus)(int)((IUniTaskSource)this).GetStatus(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
void System.Threading.Tasks.Sources.IValueTaskSource.GetResult(short token)
|
||||||
|
{
|
||||||
|
((IUniTaskSource)this).GetResult(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
void System.Threading.Tasks.Sources.IValueTaskSource.OnCompleted(Action<object> continuation, object state, short token, System.Threading.Tasks.Sources.ValueTaskSourceOnCompletedFlags flags)
|
||||||
|
{
|
||||||
|
// ignore flags, always none.
|
||||||
|
((IUniTaskSource)this).OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
public interface IUniTaskSource<out T> : IUniTaskSource
|
public interface IUniTaskSource<out T> : IUniTaskSource
|
||||||
|
#if !UNITY_2018_3_OR_NEWER && !NETSTANDARD2_0
|
||||||
|
, System.Threading.Tasks.Sources.IValueTaskSource<T>
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
new T GetResult(short token);
|
new T GetResult(short token);
|
||||||
|
|
||||||
|
#if !UNITY_2018_3_OR_NEWER && !NETSTANDARD2_0
|
||||||
|
|
||||||
|
new public UniTaskStatus GetStatus(short token)
|
||||||
|
{
|
||||||
|
return ((IUniTaskSource)this).GetStatus(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
new public void OnCompleted(Action<object> continuation, object state, short token)
|
||||||
|
{
|
||||||
|
((IUniTaskSource)this).OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
System.Threading.Tasks.Sources.ValueTaskSourceStatus System.Threading.Tasks.Sources.IValueTaskSource<T>.GetStatus(short token)
|
||||||
|
{
|
||||||
|
return (System.Threading.Tasks.Sources.ValueTaskSourceStatus)(int)((IUniTaskSource)this).GetStatus(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
T System.Threading.Tasks.Sources.IValueTaskSource<T>.GetResult(short token)
|
||||||
|
{
|
||||||
|
return ((IUniTaskSource<T>)this).GetResult(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
void System.Threading.Tasks.Sources.IValueTaskSource<T>.OnCompleted(Action<object> continuation, object state, short token, System.Threading.Tasks.Sources.ValueTaskSourceOnCompletedFlags flags)
|
||||||
|
{
|
||||||
|
// ignore flags, always none.
|
||||||
|
((IUniTaskSource)this).OnCompleted(continuation, state, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class UniTaskStatusExtensions
|
public static class UniTaskStatusExtensions
|
||||||
|
|||||||
@@ -10,6 +10,8 @@ namespace Cysharp.Threading.Tasks.Internal
|
|||||||
const int MaxArrayLength = 0X7FEFFFFF;
|
const int MaxArrayLength = 0X7FEFFFFF;
|
||||||
const int InitialSize = 16;
|
const int InitialSize = 16;
|
||||||
|
|
||||||
|
readonly PlayerLoopTiming timing;
|
||||||
|
|
||||||
SpinLock gate = new SpinLock();
|
SpinLock gate = new SpinLock();
|
||||||
bool dequing = false;
|
bool dequing = false;
|
||||||
|
|
||||||
@@ -19,6 +21,11 @@ namespace Cysharp.Threading.Tasks.Internal
|
|||||||
int waitingListCount = 0;
|
int waitingListCount = 0;
|
||||||
Action[] waitingList = new Action[InitialSize];
|
Action[] waitingList = new Action[InitialSize];
|
||||||
|
|
||||||
|
public ContinuationQueue(PlayerLoopTiming timing)
|
||||||
|
{
|
||||||
|
this.timing = timing;
|
||||||
|
}
|
||||||
|
|
||||||
public void Enqueue(Action continuation)
|
public void Enqueue(Action continuation)
|
||||||
{
|
{
|
||||||
bool lockTaken = false;
|
bool lockTaken = false;
|
||||||
@@ -72,7 +79,80 @@ namespace Cysharp.Threading.Tasks.Internal
|
|||||||
waitingList = new Action[InitialSize];
|
waitingList = new Action[InitialSize];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// delegate entrypoint.
|
||||||
public void Run()
|
public void Run()
|
||||||
|
{
|
||||||
|
// for debugging, create named stacktrace.
|
||||||
|
#if DEBUG
|
||||||
|
switch (timing)
|
||||||
|
{
|
||||||
|
case PlayerLoopTiming.Initialization:
|
||||||
|
Initialization();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastInitialization:
|
||||||
|
LastInitialization();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.EarlyUpdate:
|
||||||
|
EarlyUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastEarlyUpdate:
|
||||||
|
LastEarlyUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.FixedUpdate:
|
||||||
|
FixedUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastFixedUpdate:
|
||||||
|
LastFixedUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.PreUpdate:
|
||||||
|
PreUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastPreUpdate:
|
||||||
|
LastPreUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.Update:
|
||||||
|
Update();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastUpdate:
|
||||||
|
LastUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.PreLateUpdate:
|
||||||
|
PreLateUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastPreLateUpdate:
|
||||||
|
LastPreLateUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.PostLateUpdate:
|
||||||
|
PostLateUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastPostLateUpdate:
|
||||||
|
LastPostLateUpdate();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
RunCore();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void Initialization() => RunCore();
|
||||||
|
void LastInitialization() => RunCore();
|
||||||
|
void EarlyUpdate() => RunCore();
|
||||||
|
void LastEarlyUpdate() => RunCore();
|
||||||
|
void FixedUpdate() => RunCore();
|
||||||
|
void LastFixedUpdate() => RunCore();
|
||||||
|
void PreUpdate() => RunCore();
|
||||||
|
void LastPreUpdate() => RunCore();
|
||||||
|
void Update() => RunCore();
|
||||||
|
void LastUpdate() => RunCore();
|
||||||
|
void PreLateUpdate() => RunCore();
|
||||||
|
void LastPreLateUpdate() => RunCore();
|
||||||
|
void PostLateUpdate() => RunCore();
|
||||||
|
void LastPostLateUpdate() => RunCore();
|
||||||
|
|
||||||
|
[System.Diagnostics.DebuggerHidden]
|
||||||
|
void RunCore()
|
||||||
{
|
{
|
||||||
{
|
{
|
||||||
bool lockTaken = false;
|
bool lockTaken = false;
|
||||||
|
|||||||
@@ -239,7 +239,7 @@ namespace Cysharp.Threading.Tasks.Internal
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
var fname = fi.FullName.Replace(Path.DirectorySeparatorChar, '/').Replace(Application.dataPath, "");
|
var fname = fi.FullName.Replace(Path.DirectorySeparatorChar, '/').Replace(PlayerLoopHelper.ApplicationDataPath, "");
|
||||||
var withAssetsPath = "Assets/" + fname;
|
var withAssetsPath = "Assets/" + fname;
|
||||||
return "<a href=\"" + withAssetsPath + "\" line=\"" + line + "\">" + withAssetsPath + ":" + line + "</a>";
|
return "<a href=\"" + withAssetsPath + "\" line=\"" + line + "\">" + withAssetsPath + ":" + line + "</a>";
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,23 +0,0 @@
|
|||||||
#if NET_4_6 || NET_STANDARD_2_0 || CSHARP_7_OR_LATER
|
|
||||||
|
|
||||||
using System;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks.Internal
|
|
||||||
{
|
|
||||||
internal static class FuncExtensions
|
|
||||||
{
|
|
||||||
// avoid lambda capture
|
|
||||||
|
|
||||||
internal static Action<T> AsFuncOfT<T>(this Action action)
|
|
||||||
{
|
|
||||||
return new Action<T>(action.Invoke);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void Invoke<T>(this Action action, T unused)
|
|
||||||
{
|
|
||||||
action();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
@@ -8,6 +8,7 @@ namespace Cysharp.Threading.Tasks.Internal
|
|||||||
{
|
{
|
||||||
const int InitialSize = 16;
|
const int InitialSize = 16;
|
||||||
|
|
||||||
|
readonly PlayerLoopTiming timing;
|
||||||
readonly object runningAndQueueLock = new object();
|
readonly object runningAndQueueLock = new object();
|
||||||
readonly object arrayLock = new object();
|
readonly object arrayLock = new object();
|
||||||
readonly Action<Exception> unhandledExceptionCallback;
|
readonly Action<Exception> unhandledExceptionCallback;
|
||||||
@@ -17,9 +18,12 @@ namespace Cysharp.Threading.Tasks.Internal
|
|||||||
IPlayerLoopItem[] loopItems = new IPlayerLoopItem[InitialSize];
|
IPlayerLoopItem[] loopItems = new IPlayerLoopItem[InitialSize];
|
||||||
MinimumQueue<IPlayerLoopItem> waitQueue = new MinimumQueue<IPlayerLoopItem>(InitialSize);
|
MinimumQueue<IPlayerLoopItem> waitQueue = new MinimumQueue<IPlayerLoopItem>(InitialSize);
|
||||||
|
|
||||||
public PlayerLoopRunner()
|
|
||||||
|
|
||||||
|
public PlayerLoopRunner(PlayerLoopTiming timing)
|
||||||
{
|
{
|
||||||
this.unhandledExceptionCallback = ex => Debug.LogException(ex);
|
this.unhandledExceptionCallback = ex => Debug.LogException(ex);
|
||||||
|
this.timing = timing;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void AddAction(IPlayerLoopItem item)
|
public void AddAction(IPlayerLoopItem item)
|
||||||
@@ -55,7 +59,80 @@ namespace Cysharp.Threading.Tasks.Internal
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// delegate entrypoint.
|
||||||
public void Run()
|
public void Run()
|
||||||
|
{
|
||||||
|
// for debugging, create named stacktrace.
|
||||||
|
#if DEBUG
|
||||||
|
switch (timing)
|
||||||
|
{
|
||||||
|
case PlayerLoopTiming.Initialization:
|
||||||
|
Initialization();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastInitialization:
|
||||||
|
LastInitialization();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.EarlyUpdate:
|
||||||
|
EarlyUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastEarlyUpdate:
|
||||||
|
LastEarlyUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.FixedUpdate:
|
||||||
|
FixedUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastFixedUpdate:
|
||||||
|
LastFixedUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.PreUpdate:
|
||||||
|
PreUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastPreUpdate:
|
||||||
|
LastPreUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.Update:
|
||||||
|
Update();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastUpdate:
|
||||||
|
LastUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.PreLateUpdate:
|
||||||
|
PreLateUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastPreLateUpdate:
|
||||||
|
LastPreLateUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.PostLateUpdate:
|
||||||
|
PostLateUpdate();
|
||||||
|
break;
|
||||||
|
case PlayerLoopTiming.LastPostLateUpdate:
|
||||||
|
LastPostLateUpdate();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
RunCore();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void Initialization() => RunCore();
|
||||||
|
void LastInitialization() => RunCore();
|
||||||
|
void EarlyUpdate() => RunCore();
|
||||||
|
void LastEarlyUpdate() => RunCore();
|
||||||
|
void FixedUpdate() => RunCore();
|
||||||
|
void LastFixedUpdate() => RunCore();
|
||||||
|
void PreUpdate() => RunCore();
|
||||||
|
void LastPreUpdate() => RunCore();
|
||||||
|
void Update() => RunCore();
|
||||||
|
void LastUpdate() => RunCore();
|
||||||
|
void PreLateUpdate() => RunCore();
|
||||||
|
void LastPreLateUpdate() => RunCore();
|
||||||
|
void PostLateUpdate() => RunCore();
|
||||||
|
void LastPostLateUpdate() => RunCore();
|
||||||
|
|
||||||
|
[System.Diagnostics.DebuggerHidden]
|
||||||
|
void RunCore()
|
||||||
{
|
{
|
||||||
lock (runningAndQueueLock)
|
lock (runningAndQueueLock)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -0,0 +1,49 @@
|
|||||||
|
using System;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks.Internal
|
||||||
|
{
|
||||||
|
internal sealed class PooledDelegate<T> : ITaskPoolNode<PooledDelegate<T>>
|
||||||
|
{
|
||||||
|
static TaskPool<PooledDelegate<T>> pool;
|
||||||
|
|
||||||
|
public PooledDelegate<T> NextNode { get; set; }
|
||||||
|
|
||||||
|
static PooledDelegate()
|
||||||
|
{
|
||||||
|
TaskPool.RegisterSizeGetter(typeof(PooledDelegate<T>), () => pool.Size);
|
||||||
|
}
|
||||||
|
|
||||||
|
readonly Action<T> runDelegate;
|
||||||
|
Action continuation;
|
||||||
|
|
||||||
|
PooledDelegate()
|
||||||
|
{
|
||||||
|
runDelegate = Run;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public static Action<T> Create(Action continuation)
|
||||||
|
{
|
||||||
|
if (!pool.TryPop(out var item))
|
||||||
|
{
|
||||||
|
item = new PooledDelegate<T>();
|
||||||
|
}
|
||||||
|
|
||||||
|
item.continuation = continuation;
|
||||||
|
return item.runDelegate;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
void Run(T _)
|
||||||
|
{
|
||||||
|
var call = continuation;
|
||||||
|
continuation = null;
|
||||||
|
if (call != null)
|
||||||
|
{
|
||||||
|
pool.TryPush(this);
|
||||||
|
call.Invoke();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
fileFormatVersion: 2
|
||||||
|
guid: 8932579438742fa40b010edd412dbfba
|
||||||
|
MonoImporter:
|
||||||
|
externalObjects: {}
|
||||||
|
serializedVersion: 2
|
||||||
|
defaultReferences: []
|
||||||
|
executionOrder: 0
|
||||||
|
icon: {instanceID: 0}
|
||||||
|
userData:
|
||||||
|
assetBundleName:
|
||||||
|
assetBundleVariant:
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
using System.Collections.Concurrent;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace Cysharp.Threading.Tasks.Internal
|
|
||||||
{
|
|
||||||
// public, allow to user create custom operator with pool.
|
|
||||||
|
|
||||||
public interface IPromisePoolItem
|
|
||||||
{
|
|
||||||
void Reset();
|
|
||||||
}
|
|
||||||
|
|
||||||
public class PromisePool<T>
|
|
||||||
where T : class, IPromisePoolItem
|
|
||||||
{
|
|
||||||
int count = 0;
|
|
||||||
readonly ConcurrentQueue<T> queue = new ConcurrentQueue<T>();
|
|
||||||
readonly int maxSize;
|
|
||||||
|
|
||||||
public PromisePool(int maxSize = 256)
|
|
||||||
{
|
|
||||||
this.maxSize = maxSize;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public T TryRent()
|
|
||||||
{
|
|
||||||
if (queue.TryDequeue(out var value))
|
|
||||||
{
|
|
||||||
Interlocked.Decrement(ref count);
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public bool TryReturn(T value)
|
|
||||||
{
|
|
||||||
value.Reset(); // reset when return.
|
|
||||||
|
|
||||||
if (count < maxSize)
|
|
||||||
{
|
|
||||||
queue.Enqueue(value);
|
|
||||||
Interlocked.Increment(ref count);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -3,6 +3,7 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
|
using System.Text;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using Cysharp.Threading.Tasks.Internal;
|
using Cysharp.Threading.Tasks.Internal;
|
||||||
|
|
||||||
@@ -59,9 +60,9 @@ namespace Cysharp.Threading.Tasks
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
static List<KeyValuePair<IUniTaskSource, (int trackingId, DateTime addTime, string stackTrace)>> listPool = new List<KeyValuePair<IUniTaskSource, (int trackingId, DateTime addTime, string stackTrace)>>();
|
static List<KeyValuePair<IUniTaskSource, (string formattedType, int trackingId, DateTime addTime, string stackTrace)>> listPool = new List<KeyValuePair<IUniTaskSource, (string formattedType, int trackingId, DateTime addTime, string stackTrace)>>();
|
||||||
|
|
||||||
static readonly WeakDictionary<IUniTaskSource, (int trackingId, DateTime addTime, string stackTrace)> tracking = new WeakDictionary<IUniTaskSource, (int trackingId, DateTime addTime, string stackTrace)>();
|
static readonly WeakDictionary<IUniTaskSource, (string formattedType, int trackingId, DateTime addTime, string stackTrace)> tracking = new WeakDictionary<IUniTaskSource, (string formattedType, int trackingId, DateTime addTime, string stackTrace)>();
|
||||||
|
|
||||||
[Conditional("UNITY_EDITOR")]
|
[Conditional("UNITY_EDITOR")]
|
||||||
public static void TrackActiveTask(IUniTaskSource task, int skipFrame)
|
public static void TrackActiveTask(IUniTaskSource task, int skipFrame)
|
||||||
@@ -70,7 +71,19 @@ namespace Cysharp.Threading.Tasks
|
|||||||
dirty = true;
|
dirty = true;
|
||||||
if (!EditorEnableState.EnableTracking) return;
|
if (!EditorEnableState.EnableTracking) return;
|
||||||
var stackTrace = EditorEnableState.EnableStackTrace ? new StackTrace(skipFrame, true).CleanupAsyncStackTrace() : "";
|
var stackTrace = EditorEnableState.EnableStackTrace ? new StackTrace(skipFrame, true).CleanupAsyncStackTrace() : "";
|
||||||
tracking.TryAdd(task, (Interlocked.Increment(ref trackingId), DateTime.UtcNow, stackTrace));
|
|
||||||
|
string typeName;
|
||||||
|
if (EditorEnableState.EnableStackTrace)
|
||||||
|
{
|
||||||
|
var sb = new StringBuilder();
|
||||||
|
TypeBeautify(task.GetType(), sb);
|
||||||
|
typeName = sb.ToString();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
typeName = task.GetType().Name;
|
||||||
|
}
|
||||||
|
tracking.TryAdd(task, (typeName, Interlocked.Increment(ref trackingId), DateTime.UtcNow, stackTrace));
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -103,19 +116,8 @@ namespace Cysharp.Threading.Tasks
|
|||||||
{
|
{
|
||||||
for (int i = 0; i < count; i++)
|
for (int i = 0; i < count; i++)
|
||||||
{
|
{
|
||||||
string typeName = null;
|
action(listPool[i].Value.trackingId, listPool[i].Value.formattedType, listPool[i].Key.UnsafeGetStatus(), listPool[i].Value.addTime, listPool[i].Value.stackTrace);
|
||||||
var keyType = listPool[i].Key.GetType();
|
listPool[i] = default;
|
||||||
if (keyType.IsNested)
|
|
||||||
{
|
|
||||||
typeName = keyType.DeclaringType.Name + "." + keyType.Name;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
typeName = keyType.Name;
|
|
||||||
}
|
|
||||||
|
|
||||||
action(listPool[i].Value.trackingId, typeName, listPool[i].Key.UnsafeGetStatus(), listPool[i].Value.addTime, listPool[i].Value.stackTrace);
|
|
||||||
listPool[i] = new KeyValuePair<IUniTaskSource, (int trackingId, DateTime addTime, string stackTrace)>(null, (0, default(DateTime), null)); // clear
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch
|
catch
|
||||||
@@ -125,6 +127,52 @@ namespace Cysharp.Threading.Tasks
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void TypeBeautify(Type type, StringBuilder sb)
|
||||||
|
{
|
||||||
|
if (type.IsNested)
|
||||||
|
{
|
||||||
|
// TypeBeautify(type.DeclaringType, sb);
|
||||||
|
sb.Append(type.DeclaringType.Name.ToString());
|
||||||
|
sb.Append(".");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type.IsGenericType)
|
||||||
|
{
|
||||||
|
var genericsStart = type.Name.IndexOf("`");
|
||||||
|
if (genericsStart != -1)
|
||||||
|
{
|
||||||
|
sb.Append(type.Name.Substring(0, genericsStart));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sb.Append(type.Name);
|
||||||
|
}
|
||||||
|
sb.Append("<");
|
||||||
|
var first = true;
|
||||||
|
foreach (var item in type.GetGenericArguments())
|
||||||
|
{
|
||||||
|
if (!first)
|
||||||
|
{
|
||||||
|
sb.Append(", ");
|
||||||
|
}
|
||||||
|
first = false;
|
||||||
|
TypeBeautify(item, sb);
|
||||||
|
}
|
||||||
|
sb.Append(">");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sb.Append(type.Name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//static string RemoveUniTaskNamespace(string str)
|
||||||
|
//{
|
||||||
|
// return str.Replace("Cysharp.Threading.Tasks.CompilerServices", "")
|
||||||
|
// .Replace("Cysharp.Threading.Tasks.Linq", "")
|
||||||
|
// .Replace("Cysharp.Threading.Tasks", "");
|
||||||
|
//}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
||||||
|
|
||||||
return Aggregate.InvokeAsync(source, accumulator, cancellationToken);
|
return Aggregate.AggregateAsync(source, accumulator, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TAccumulate> AggregateAsync<TSource, TAccumulate>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator, CancellationToken cancellationToken = default)
|
public static UniTask<TAccumulate> AggregateAsync<TSource, TAccumulate>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator, CancellationToken cancellationToken = default)
|
||||||
@@ -20,7 +20,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
||||||
|
|
||||||
return Aggregate.InvokeAsync(source, seed, accumulator, cancellationToken);
|
return Aggregate.AggregateAsync(source, seed, accumulator, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TResult> AggregateAsync<TSource, TAccumulate, TResult>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator, Func<TAccumulate, TResult> resultSelector, CancellationToken cancellationToken = default)
|
public static UniTask<TResult> AggregateAsync<TSource, TAccumulate, TResult>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator, Func<TAccumulate, TResult> resultSelector, CancellationToken cancellationToken = default)
|
||||||
@@ -29,7 +29,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
||||||
Error.ThrowArgumentNullException(accumulator, nameof(resultSelector));
|
Error.ThrowArgumentNullException(accumulator, nameof(resultSelector));
|
||||||
|
|
||||||
return Aggregate.InvokeAsync(source, seed, accumulator, resultSelector, cancellationToken);
|
return Aggregate.AggregateAsync(source, seed, accumulator, resultSelector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> AggregateAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TSource, UniTask<TSource>> accumulator, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> AggregateAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TSource, UniTask<TSource>> accumulator, CancellationToken cancellationToken = default)
|
||||||
@@ -37,7 +37,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
||||||
|
|
||||||
return Aggregate.InvokeAsync(source, accumulator, cancellationToken);
|
return Aggregate.AggregateAwaitAsync(source, accumulator, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TAccumulate> AggregateAwaitAsync<TSource, TAccumulate>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, UniTask<TAccumulate>> accumulator, CancellationToken cancellationToken = default)
|
public static UniTask<TAccumulate> AggregateAwaitAsync<TSource, TAccumulate>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, UniTask<TAccumulate>> accumulator, CancellationToken cancellationToken = default)
|
||||||
@@ -45,7 +45,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
||||||
|
|
||||||
return Aggregate.InvokeAsync(source, seed, accumulator, cancellationToken);
|
return Aggregate.AggregateAwaitAsync(source, seed, accumulator, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TResult> AggregateAwaitAsync<TSource, TAccumulate, TResult>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, UniTask<TAccumulate>> accumulator, Func<TAccumulate, UniTask<TResult>> resultSelector, CancellationToken cancellationToken = default)
|
public static UniTask<TResult> AggregateAwaitAsync<TSource, TAccumulate, TResult>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, UniTask<TAccumulate>> accumulator, Func<TAccumulate, UniTask<TResult>> resultSelector, CancellationToken cancellationToken = default)
|
||||||
@@ -54,7 +54,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
||||||
Error.ThrowArgumentNullException(accumulator, nameof(resultSelector));
|
Error.ThrowArgumentNullException(accumulator, nameof(resultSelector));
|
||||||
|
|
||||||
return Aggregate.InvokeAsync(source, seed, accumulator, resultSelector, cancellationToken);
|
return Aggregate.AggregateAwaitAsync(source, seed, accumulator, resultSelector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> AggregateAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TSource, CancellationToken, UniTask<TSource>> accumulator, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> AggregateAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TSource, CancellationToken, UniTask<TSource>> accumulator, CancellationToken cancellationToken = default)
|
||||||
@@ -62,7 +62,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
||||||
|
|
||||||
return Aggregate.InvokeAsync(source, accumulator, cancellationToken);
|
return Aggregate.AggregateAwaitWithCancellationAsync(source, accumulator, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TAccumulate> AggregateAwaitWithCancellationAsync<TSource, TAccumulate>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, CancellationToken, UniTask<TAccumulate>> accumulator, CancellationToken cancellationToken = default)
|
public static UniTask<TAccumulate> AggregateAwaitWithCancellationAsync<TSource, TAccumulate>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, CancellationToken, UniTask<TAccumulate>> accumulator, CancellationToken cancellationToken = default)
|
||||||
@@ -70,7 +70,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
||||||
|
|
||||||
return Aggregate.InvokeAsync(source, seed, accumulator, cancellationToken);
|
return Aggregate.AggregateAwaitWithCancellationAsync(source, seed, accumulator, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TResult> AggregateAwaitWithCancellationAsync<TSource, TAccumulate, TResult>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, CancellationToken, UniTask<TAccumulate>> accumulator, Func<TAccumulate, CancellationToken, UniTask<TResult>> resultSelector, CancellationToken cancellationToken = default)
|
public static UniTask<TResult> AggregateAwaitWithCancellationAsync<TSource, TAccumulate, TResult>(this IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, CancellationToken, UniTask<TAccumulate>> accumulator, Func<TAccumulate, CancellationToken, UniTask<TResult>> resultSelector, CancellationToken cancellationToken = default)
|
||||||
@@ -79,13 +79,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
Error.ThrowArgumentNullException(accumulator, nameof(accumulator));
|
||||||
Error.ThrowArgumentNullException(accumulator, nameof(resultSelector));
|
Error.ThrowArgumentNullException(accumulator, nameof(resultSelector));
|
||||||
|
|
||||||
return Aggregate.InvokeAsync(source, seed, accumulator, resultSelector, cancellationToken);
|
return Aggregate.AggregateAwaitWithCancellationAsync(source, seed, accumulator, resultSelector, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static class Aggregate
|
internal static class Aggregate
|
||||||
{
|
{
|
||||||
internal static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TSource, TSource> accumulator, CancellationToken cancellationToken)
|
internal static async UniTask<TSource> AggregateAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TSource, TSource> accumulator, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -116,7 +116,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<TAccumulate> InvokeAsync<TSource, TAccumulate>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator, CancellationToken cancellationToken)
|
internal static async UniTask<TAccumulate> AggregateAsync<TSource, TAccumulate>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -138,7 +138,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<TResult> InvokeAsync<TSource, TAccumulate, TResult>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator, Func<TAccumulate, TResult> resultSelector, CancellationToken cancellationToken)
|
internal static async UniTask<TResult> AggregateAsync<TSource, TAccumulate, TResult>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator, Func<TAccumulate, TResult> resultSelector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -162,7 +162,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
// with async
|
// with async
|
||||||
|
|
||||||
internal static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TSource, UniTask<TSource>> accumulator, CancellationToken cancellationToken)
|
internal static async UniTask<TSource> AggregateAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TSource, UniTask<TSource>> accumulator, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -193,7 +193,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<TAccumulate> InvokeAsync<TSource, TAccumulate>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, UniTask<TAccumulate>> accumulator, CancellationToken cancellationToken)
|
internal static async UniTask<TAccumulate> AggregateAwaitAsync<TSource, TAccumulate>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, UniTask<TAccumulate>> accumulator, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -215,7 +215,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<TResult> InvokeAsync<TSource, TAccumulate, TResult>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, UniTask<TAccumulate>> accumulator, Func<TAccumulate, UniTask<TResult>> resultSelector, CancellationToken cancellationToken)
|
internal static async UniTask<TResult> AggregateAwaitAsync<TSource, TAccumulate, TResult>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, UniTask<TAccumulate>> accumulator, Func<TAccumulate, UniTask<TResult>> resultSelector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -240,7 +240,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
// with cancellation
|
// with cancellation
|
||||||
|
|
||||||
internal static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TSource, CancellationToken, UniTask<TSource>> accumulator, CancellationToken cancellationToken)
|
internal static async UniTask<TSource> AggregateAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TSource, CancellationToken, UniTask<TSource>> accumulator, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -271,7 +271,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<TAccumulate> InvokeAsync<TSource, TAccumulate>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, CancellationToken, UniTask<TAccumulate>> accumulator, CancellationToken cancellationToken)
|
internal static async UniTask<TAccumulate> AggregateAwaitWithCancellationAsync<TSource, TAccumulate>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, CancellationToken, UniTask<TAccumulate>> accumulator, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -293,7 +293,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<TResult> InvokeAsync<TSource, TAccumulate, TResult>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, CancellationToken, UniTask<TAccumulate>> accumulator, Func<TAccumulate, CancellationToken, UniTask<TResult>> resultSelector, CancellationToken cancellationToken)
|
internal static async UniTask<TResult> AggregateAwaitWithCancellationAsync<TSource, TAccumulate, TResult>(IUniTaskAsyncEnumerable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, CancellationToken, UniTask<TAccumulate>> accumulator, Func<TAccumulate, CancellationToken, UniTask<TResult>> resultSelector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return All.InvokeAsync(source, predicate, cancellationToken);
|
return All.AllAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<Boolean> AllAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<Boolean> AllAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -19,7 +19,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return All.InvokeAsync(source, predicate, cancellationToken);
|
return All.AllAwaitAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<Boolean> AllAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<Boolean> AllAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -27,13 +27,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return All.InvokeAsync(source, predicate, cancellationToken);
|
return All.AllAwaitWithCancellationAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static class All
|
internal static class All
|
||||||
{
|
{
|
||||||
internal static async UniTask<bool> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<bool> AllAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -57,7 +57,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<bool> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<bool> AllAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -81,7 +81,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<bool> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<bool> AllAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Any.InvokeAsync(source, cancellationToken);
|
return Any.AnyAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<Boolean> AnyAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<Boolean> AnyAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -18,7 +18,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Any.InvokeAsync(source, predicate, cancellationToken);
|
return Any.AnyAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<Boolean> AnyAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<Boolean> AnyAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -26,7 +26,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Any.InvokeAsync(source, predicate, cancellationToken);
|
return Any.AnyAwaitAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<Boolean> AnyAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<Boolean> AnyAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -34,13 +34,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Any.InvokeAsync(source, predicate, cancellationToken);
|
return Any.AnyAwaitWithCancellationAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static class Any
|
internal static class Any
|
||||||
{
|
{
|
||||||
internal static async UniTask<bool> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
|
internal static async UniTask<bool> AnyAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -61,7 +61,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<bool> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<bool> AnyAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -85,7 +85,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<bool> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<bool> AnyAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -109,7 +109,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<bool> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<bool> AnyAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
|
|||||||
@@ -36,10 +36,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, element, append, cancellationToken);
|
return new _AppendPrepend(source, element, append, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
sealed class _AppendPrepend : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
||||||
{
|
{
|
||||||
enum State : byte
|
enum State : byte
|
||||||
{
|
{
|
||||||
@@ -59,12 +59,14 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
IUniTaskAsyncEnumerator<TSource> enumerator;
|
IUniTaskAsyncEnumerator<TSource> enumerator;
|
||||||
UniTask<bool>.Awaiter awaiter;
|
UniTask<bool>.Awaiter awaiter;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, TSource element, bool append, CancellationToken cancellationToken)
|
public _AppendPrepend(IUniTaskAsyncEnumerable<TSource> source, TSource element, bool append, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.element = element;
|
this.element = element;
|
||||||
this.state = append ? State.RequireAppend : State.RequirePrepend;
|
this.state = append ? State.RequireAppend : State.RequirePrepend;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TSource Current { get; private set; }
|
public TSource Current { get; private set; }
|
||||||
@@ -108,7 +110,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_AppendPrepend)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -136,6 +138,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (enumerator != null)
|
if (enumerator != null)
|
||||||
{
|
{
|
||||||
return enumerator.DisposeAsync();
|
return enumerator.DisposeAsync();
|
||||||
|
|||||||
@@ -3,66 +3,9 @@ using System.Threading;
|
|||||||
|
|
||||||
namespace Cysharp.Threading.Tasks.Linq
|
namespace Cysharp.Threading.Tasks.Linq
|
||||||
{
|
{
|
||||||
public abstract class MoveNextSource : IUniTaskSource<bool>
|
// note: refactor all inherit class and should remove this.
|
||||||
{
|
// see Select and Where.
|
||||||
protected UniTaskCompletionSourceCore<bool> completionSource;
|
internal abstract class AsyncEnumeratorBase<TSource, TResult> : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
|
|
||||||
public bool GetResult(short token)
|
|
||||||
{
|
|
||||||
return completionSource.GetResult(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus GetStatus(short token)
|
|
||||||
{
|
|
||||||
return completionSource.GetStatus(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnCompleted(Action<object> continuation, object state, short token)
|
|
||||||
{
|
|
||||||
completionSource.OnCompleted(continuation, state, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UniTaskStatus UnsafeGetStatus()
|
|
||||||
{
|
|
||||||
return completionSource.UnsafeGetStatus();
|
|
||||||
}
|
|
||||||
|
|
||||||
void IUniTaskSource.GetResult(short token)
|
|
||||||
{
|
|
||||||
completionSource.GetResult(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected bool TryGetResult<T>(UniTask<T>.Awaiter awaiter, out T result)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
result = awaiter.GetResult();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
completionSource.TrySetException(ex);
|
|
||||||
result = default;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected bool TryGetResult(UniTask.Awaiter awaiter)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
awaiter.GetResult();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
completionSource.TrySetException(ex);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public abstract class AsyncEnumeratorBase<TSource, TResult> : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
|
||||||
{
|
{
|
||||||
static readonly Action<object> moveNextCallbackDelegate = MoveNextCallBack;
|
static readonly Action<object> moveNextCallbackDelegate = MoveNextCallBack;
|
||||||
|
|
||||||
@@ -76,6 +19,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
// abstract
|
// abstract
|
||||||
@@ -178,6 +122,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
// if require additional resource to dispose, override and call base.DisposeAsync.
|
// if require additional resource to dispose, override and call base.DisposeAsync.
|
||||||
public virtual UniTask DisposeAsync()
|
public virtual UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (enumerator != null)
|
if (enumerator != null)
|
||||||
{
|
{
|
||||||
return enumerator.DisposeAsync();
|
return enumerator.DisposeAsync();
|
||||||
@@ -186,7 +131,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public abstract class AsyncEnumeratorAwaitSelectorBase<TSource, TResult, TAwait> : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
internal abstract class AsyncEnumeratorAwaitSelectorBase<TSource, TResult, TAwait> : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
{
|
{
|
||||||
static readonly Action<object> moveNextCallbackDelegate = MoveNextCallBack;
|
static readonly Action<object> moveNextCallbackDelegate = MoveNextCallBack;
|
||||||
static readonly Action<object> setCurrentCallbackDelegate = SetCurrentCallBack;
|
static readonly Action<object> setCurrentCallbackDelegate = SetCurrentCallBack;
|
||||||
@@ -204,6 +149,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
// abstract
|
// abstract
|
||||||
@@ -399,6 +345,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
// if require additional resource to dispose, override and call base.DisposeAsync.
|
// if require additional resource to dispose, override and call base.DisposeAsync.
|
||||||
public virtual UniTask DisposeAsync()
|
public virtual UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (enumerator != null)
|
if (enumerator != null)
|
||||||
{
|
{
|
||||||
return enumerator.DisposeAsync();
|
return enumerator.DisposeAsync();
|
||||||
@@ -406,5 +353,4 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return default;
|
return default;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -10,7 +10,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, cancellationToken);
|
return Average.AverageAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -18,7 +18,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int32>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int32>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -26,22 +26,22 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double> AverageAwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int32>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double> AverageAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int32>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double> AverageAsync(this IUniTaskAsyncEnumerable<Int64> source, CancellationToken cancellationToken = default)
|
public static UniTask<double> AverageAsync(this IUniTaskAsyncEnumerable<Int64> source, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, cancellationToken);
|
return Average.AverageAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int64> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int64> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -49,7 +49,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int64>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int64>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -57,22 +57,22 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double> AverageAwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int64>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double> AverageAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int64>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<float> AverageAsync(this IUniTaskAsyncEnumerable<Single> source, CancellationToken cancellationToken = default)
|
public static UniTask<float> AverageAsync(this IUniTaskAsyncEnumerable<Single> source, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, cancellationToken);
|
return Average.AverageAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<float> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Single> selector, CancellationToken cancellationToken = default)
|
public static UniTask<float> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Single> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -80,7 +80,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<float> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Single>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<float> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Single>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -88,22 +88,22 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<float> AverageAwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Single>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<float> AverageAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Single>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double> AverageAsync(this IUniTaskAsyncEnumerable<Double> source, CancellationToken cancellationToken = default)
|
public static UniTask<double> AverageAsync(this IUniTaskAsyncEnumerable<Double> source, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, cancellationToken);
|
return Average.AverageAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Double> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Double> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -111,7 +111,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Double>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Double>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -119,22 +119,22 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double> AverageAwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Double>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double> AverageAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Double>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<decimal> AverageAsync(this IUniTaskAsyncEnumerable<Decimal> source, CancellationToken cancellationToken = default)
|
public static UniTask<decimal> AverageAsync(this IUniTaskAsyncEnumerable<Decimal> source, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, cancellationToken);
|
return Average.AverageAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<decimal> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Decimal> selector, CancellationToken cancellationToken = default)
|
public static UniTask<decimal> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Decimal> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -142,7 +142,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<decimal> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Decimal>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<decimal> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Decimal>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -150,22 +150,22 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<decimal> AverageAwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Decimal>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<decimal> AverageAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Decimal>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAsync(this IUniTaskAsyncEnumerable<Int32?> source, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAsync(this IUniTaskAsyncEnumerable<Int32?> source, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, cancellationToken);
|
return Average.AverageAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32?> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32?> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -173,7 +173,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int32?>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int32?>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -181,22 +181,22 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int32?>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int32?>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAsync(this IUniTaskAsyncEnumerable<Int64?> source, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAsync(this IUniTaskAsyncEnumerable<Int64?> source, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, cancellationToken);
|
return Average.AverageAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int64?> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int64?> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -204,7 +204,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int64?>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int64?>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -212,22 +212,22 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int64?>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int64?>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<float?> AverageAsync(this IUniTaskAsyncEnumerable<Single?> source, CancellationToken cancellationToken = default)
|
public static UniTask<float?> AverageAsync(this IUniTaskAsyncEnumerable<Single?> source, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, cancellationToken);
|
return Average.AverageAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<float?> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Single?> selector, CancellationToken cancellationToken = default)
|
public static UniTask<float?> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Single?> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -235,7 +235,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<float?> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Single?>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<float?> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Single?>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -243,22 +243,22 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<float?> AverageAwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Single?>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<float?> AverageAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Single?>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAsync(this IUniTaskAsyncEnumerable<Double?> source, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAsync(this IUniTaskAsyncEnumerable<Double?> source, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, cancellationToken);
|
return Average.AverageAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Double?> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Double?> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -266,7 +266,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Double?>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Double?>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -274,22 +274,22 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<double?> AverageAwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Double?>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<double?> AverageAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Double?>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<decimal?> AverageAsync(this IUniTaskAsyncEnumerable<Decimal?> source, CancellationToken cancellationToken = default)
|
public static UniTask<decimal?> AverageAsync(this IUniTaskAsyncEnumerable<Decimal?> source, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, cancellationToken);
|
return Average.AverageAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<decimal?> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Decimal?> selector, CancellationToken cancellationToken = default)
|
public static UniTask<decimal?> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Decimal?> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -297,7 +297,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<decimal?> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Decimal?>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<decimal?> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Decimal?>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -305,22 +305,22 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<decimal?> AverageAwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Decimal?>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<decimal?> AverageAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Decimal?>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static class Average
|
internal static class Average
|
||||||
{
|
{
|
||||||
public static async UniTask<double> InvokeAsync(IUniTaskAsyncEnumerable<Int32> source, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAsync(IUniTaskAsyncEnumerable<Int32> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int32 sum = 0;
|
Int32 sum = 0;
|
||||||
@@ -348,7 +348,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32> selector, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int32 sum = 0;
|
Int32 sum = 0;
|
||||||
@@ -376,7 +376,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int32>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int32>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int32 sum = 0;
|
Int32 sum = 0;
|
||||||
@@ -404,7 +404,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int32>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int32>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int32 sum = 0;
|
Int32 sum = 0;
|
||||||
@@ -432,7 +432,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double> InvokeAsync(IUniTaskAsyncEnumerable<Int64> source, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAsync(IUniTaskAsyncEnumerable<Int64> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int64 sum = 0;
|
Int64 sum = 0;
|
||||||
@@ -460,7 +460,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int64> selector, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int64> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int64 sum = 0;
|
Int64 sum = 0;
|
||||||
@@ -488,7 +488,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int64>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int64>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int64 sum = 0;
|
Int64 sum = 0;
|
||||||
@@ -516,7 +516,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int64>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int64>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int64 sum = 0;
|
Int64 sum = 0;
|
||||||
@@ -544,7 +544,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<float> InvokeAsync(IUniTaskAsyncEnumerable<Single> source, CancellationToken cancellationToken)
|
public static async UniTask<float> AverageAsync(IUniTaskAsyncEnumerable<Single> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Single sum = 0;
|
Single sum = 0;
|
||||||
@@ -572,7 +572,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (float)(sum / count);
|
return (float)(sum / count);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<float> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Single> selector, CancellationToken cancellationToken)
|
public static async UniTask<float> AverageAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Single> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Single sum = 0;
|
Single sum = 0;
|
||||||
@@ -600,7 +600,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (float)(sum / count);
|
return (float)(sum / count);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<float> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Single>> selector, CancellationToken cancellationToken)
|
public static async UniTask<float> AverageAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Single>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Single sum = 0;
|
Single sum = 0;
|
||||||
@@ -628,7 +628,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (float)(sum / count);
|
return (float)(sum / count);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<float> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Single>> selector, CancellationToken cancellationToken)
|
public static async UniTask<float> AverageAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Single>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Single sum = 0;
|
Single sum = 0;
|
||||||
@@ -656,7 +656,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (float)(sum / count);
|
return (float)(sum / count);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double> InvokeAsync(IUniTaskAsyncEnumerable<Double> source, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAsync(IUniTaskAsyncEnumerable<Double> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Double sum = 0;
|
Double sum = 0;
|
||||||
@@ -684,7 +684,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Double> selector, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Double> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Double sum = 0;
|
Double sum = 0;
|
||||||
@@ -712,7 +712,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Double>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Double>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Double sum = 0;
|
Double sum = 0;
|
||||||
@@ -740,7 +740,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Double>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double> AverageAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Double>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Double sum = 0;
|
Double sum = 0;
|
||||||
@@ -768,7 +768,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<decimal> InvokeAsync(IUniTaskAsyncEnumerable<Decimal> source, CancellationToken cancellationToken)
|
public static async UniTask<decimal> AverageAsync(IUniTaskAsyncEnumerable<Decimal> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Decimal sum = 0;
|
Decimal sum = 0;
|
||||||
@@ -796,7 +796,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<decimal> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Decimal> selector, CancellationToken cancellationToken)
|
public static async UniTask<decimal> AverageAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Decimal> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Decimal sum = 0;
|
Decimal sum = 0;
|
||||||
@@ -824,7 +824,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<decimal> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Decimal>> selector, CancellationToken cancellationToken)
|
public static async UniTask<decimal> AverageAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Decimal>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Decimal sum = 0;
|
Decimal sum = 0;
|
||||||
@@ -852,7 +852,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<decimal> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Decimal>> selector, CancellationToken cancellationToken)
|
public static async UniTask<decimal> AverageAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Decimal>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Decimal sum = 0;
|
Decimal sum = 0;
|
||||||
@@ -880,7 +880,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync(IUniTaskAsyncEnumerable<Int32?> source, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAsync(IUniTaskAsyncEnumerable<Int32?> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int32? sum = 0;
|
Int32? sum = 0;
|
||||||
@@ -912,7 +912,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32?> selector, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32?> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int32? sum = 0;
|
Int32? sum = 0;
|
||||||
@@ -944,7 +944,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int32?>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int32?>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int32? sum = 0;
|
Int32? sum = 0;
|
||||||
@@ -976,7 +976,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int32?>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int32?>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int32? sum = 0;
|
Int32? sum = 0;
|
||||||
@@ -1008,7 +1008,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync(IUniTaskAsyncEnumerable<Int64?> source, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAsync(IUniTaskAsyncEnumerable<Int64?> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int64? sum = 0;
|
Int64? sum = 0;
|
||||||
@@ -1040,7 +1040,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int64?> selector, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int64?> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int64? sum = 0;
|
Int64? sum = 0;
|
||||||
@@ -1072,7 +1072,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int64?>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Int64?>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int64? sum = 0;
|
Int64? sum = 0;
|
||||||
@@ -1104,7 +1104,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int64?>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Int64?>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Int64? sum = 0;
|
Int64? sum = 0;
|
||||||
@@ -1136,7 +1136,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (double)sum / count;
|
return (double)sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<float?> InvokeAsync(IUniTaskAsyncEnumerable<Single?> source, CancellationToken cancellationToken)
|
public static async UniTask<float?> AverageAsync(IUniTaskAsyncEnumerable<Single?> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Single? sum = 0;
|
Single? sum = 0;
|
||||||
@@ -1168,7 +1168,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (float)(sum / count);
|
return (float)(sum / count);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<float?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Single?> selector, CancellationToken cancellationToken)
|
public static async UniTask<float?> AverageAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Single?> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Single? sum = 0;
|
Single? sum = 0;
|
||||||
@@ -1200,7 +1200,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (float)(sum / count);
|
return (float)(sum / count);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<float?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Single?>> selector, CancellationToken cancellationToken)
|
public static async UniTask<float?> AverageAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Single?>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Single? sum = 0;
|
Single? sum = 0;
|
||||||
@@ -1232,7 +1232,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (float)(sum / count);
|
return (float)(sum / count);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<float?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Single?>> selector, CancellationToken cancellationToken)
|
public static async UniTask<float?> AverageAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Single?>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Single? sum = 0;
|
Single? sum = 0;
|
||||||
@@ -1264,7 +1264,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return (float)(sum / count);
|
return (float)(sum / count);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync(IUniTaskAsyncEnumerable<Double?> source, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAsync(IUniTaskAsyncEnumerable<Double?> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Double? sum = 0;
|
Double? sum = 0;
|
||||||
@@ -1296,7 +1296,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Double?> selector, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Double?> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Double? sum = 0;
|
Double? sum = 0;
|
||||||
@@ -1328,7 +1328,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Double?>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Double?>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Double? sum = 0;
|
Double? sum = 0;
|
||||||
@@ -1360,7 +1360,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<double?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Double?>> selector, CancellationToken cancellationToken)
|
public static async UniTask<double?> AverageAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Double?>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Double? sum = 0;
|
Double? sum = 0;
|
||||||
@@ -1392,7 +1392,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<decimal?> InvokeAsync(IUniTaskAsyncEnumerable<Decimal?> source, CancellationToken cancellationToken)
|
public static async UniTask<decimal?> AverageAsync(IUniTaskAsyncEnumerable<Decimal?> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Decimal? sum = 0;
|
Decimal? sum = 0;
|
||||||
@@ -1424,7 +1424,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<decimal?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Decimal?> selector, CancellationToken cancellationToken)
|
public static async UniTask<decimal?> AverageAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Decimal?> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Decimal? sum = 0;
|
Decimal? sum = 0;
|
||||||
@@ -1456,7 +1456,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<decimal?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Decimal?>> selector, CancellationToken cancellationToken)
|
public static async UniTask<decimal?> AverageAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Decimal?>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Decimal? sum = 0;
|
Decimal? sum = 0;
|
||||||
@@ -1488,7 +1488,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return sum / count;
|
return sum / count;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<decimal?> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Decimal?>> selector, CancellationToken cancellationToken)
|
public static async UniTask<decimal?> AverageAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Decimal?>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
Decimal? sum = 0;
|
Decimal? sum = 0;
|
||||||
|
|||||||
@@ -39,7 +39,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, cancellationToken);
|
return Average.AverageAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<<#= ret #>> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, <#= TypeName(t) #>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<<#= ret #>> AverageAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, <#= TypeName(t) #>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -47,7 +47,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<<#= ret #>> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<<#= ret #>> AverageAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -55,15 +55,15 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<<#= ret #>> AverageAwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<<#= ret #>> AverageAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Average.InvokeAsync(source, selector, cancellationToken);
|
return Average.AverageAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
<# } #>
|
<# } #>
|
||||||
@@ -72,7 +72,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
internal static class Average
|
internal static class Average
|
||||||
{
|
{
|
||||||
<# foreach(var (t, ret) in types) { #>
|
<# foreach(var (t, ret) in types) { #>
|
||||||
public static async UniTask<<#= ret #>> InvokeAsync(IUniTaskAsyncEnumerable<<#= TypeName(t) #>> source, CancellationToken cancellationToken)
|
public static async UniTask<<#= ret #>> AverageAsync(IUniTaskAsyncEnumerable<<#= TypeName(t) #>> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
<#= TypeName(t) #> sum = 0;
|
<#= TypeName(t) #> sum = 0;
|
||||||
@@ -112,7 +112,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return <#= CalcResult(t) #>;
|
return <#= CalcResult(t) #>;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<<#= ret #>> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, <#= TypeName(t) #>> selector, CancellationToken cancellationToken)
|
public static async UniTask<<#= ret #>> AverageAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, <#= TypeName(t) #>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
<#= TypeName(t) #> sum = 0;
|
<#= TypeName(t) #> sum = 0;
|
||||||
@@ -152,7 +152,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return <#= CalcResult(t) #>;
|
return <#= CalcResult(t) #>;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<<#= ret #>> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken)
|
public static async UniTask<<#= ret #>> AverageAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
<#= TypeName(t) #> sum = 0;
|
<#= TypeName(t) #> sum = 0;
|
||||||
@@ -192,7 +192,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return <#= CalcResult(t) #>;
|
return <#= CalcResult(t) #>;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<<#= ret #>> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken)
|
public static async UniTask<<#= ret #>> AverageAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
<#= TypeName(t) #> sum = 0;
|
<#= TypeName(t) #> sum = 0;
|
||||||
|
|||||||
@@ -38,10 +38,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<IList<TSource>> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<IList<TSource>> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, count, cancellationToken);
|
return new _Buffer(source, count, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<IList<TSource>>
|
sealed class _Buffer : MoveNextSource, IUniTaskAsyncEnumerator<IList<TSource>>
|
||||||
{
|
{
|
||||||
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
||||||
|
|
||||||
@@ -56,11 +56,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
bool completed;
|
bool completed;
|
||||||
List<TSource> buffer;
|
List<TSource> buffer;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, int count, CancellationToken cancellationToken)
|
public _Buffer(IUniTaskAsyncEnumerable<TSource> source, int count, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.count = count;
|
this.count = count;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public IList<TSource> Current { get; private set; }
|
public IList<TSource> Current { get; private set; }
|
||||||
@@ -128,7 +130,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_Buffer)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -167,6 +169,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (enumerator != null)
|
if (enumerator != null)
|
||||||
{
|
{
|
||||||
return enumerator.DisposeAsync();
|
return enumerator.DisposeAsync();
|
||||||
@@ -191,10 +194,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<IList<TSource>> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<IList<TSource>> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, count, skip, cancellationToken);
|
return new _BufferSkip(source, count, skip, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<IList<TSource>>
|
sealed class _BufferSkip : MoveNextSource, IUniTaskAsyncEnumerator<IList<TSource>>
|
||||||
{
|
{
|
||||||
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
||||||
|
|
||||||
@@ -211,12 +214,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Queue<List<TSource>> buffers;
|
Queue<List<TSource>> buffers;
|
||||||
int index = 0;
|
int index = 0;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, int count, int skip, CancellationToken cancellationToken)
|
public _BufferSkip(IUniTaskAsyncEnumerable<TSource> source, int count, int skip, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.count = count;
|
this.count = count;
|
||||||
this.skip = skip;
|
this.skip = skip;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public IList<TSource> Current { get; private set; }
|
public IList<TSource> Current { get; private set; }
|
||||||
@@ -282,7 +286,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_BufferSkip)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -329,6 +333,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (enumerator != null)
|
if (enumerator != null)
|
||||||
{
|
{
|
||||||
return enumerator.DisposeAsync();
|
return enumerator.DisposeAsync();
|
||||||
|
|||||||
@@ -25,12 +25,12 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, cancellationToken);
|
return new _Cast(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorBase<object, TResult>
|
class _Cast : AsyncEnumeratorBase<object, TResult>
|
||||||
{
|
{
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<object> source, CancellationToken cancellationToken)
|
public _Cast(IUniTaskAsyncEnumerable<object> source, CancellationToken cancellationToken)
|
||||||
|
|
||||||
: base(source, cancellationToken)
|
: base(source, cancellationToken)
|
||||||
{
|
{
|
||||||
|
|||||||
11372
src/UniTask/Assets/Plugins/UniTask/Runtime/Linq/CombineLatest.cs
Normal file
11372
src/UniTask/Assets/Plugins/UniTask/Runtime/Linq/CombineLatest.cs
Normal file
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,11 @@
|
|||||||
|
fileFormatVersion: 2
|
||||||
|
guid: 6cb07f6e88287e34d9b9301a572284a5
|
||||||
|
MonoImporter:
|
||||||
|
externalObjects: {}
|
||||||
|
serializedVersion: 2
|
||||||
|
defaultReferences: []
|
||||||
|
executionOrder: 0
|
||||||
|
icon: {instanceID: 0}
|
||||||
|
userData:
|
||||||
|
assetBundleName:
|
||||||
|
assetBundleVariant:
|
||||||
221
src/UniTask/Assets/Plugins/UniTask/Runtime/Linq/CombineLatest.tt
Normal file
221
src/UniTask/Assets/Plugins/UniTask/Runtime/Linq/CombineLatest.tt
Normal file
@@ -0,0 +1,221 @@
|
|||||||
|
<#@ template debug="false" hostspecific="false" language="C#" #>
|
||||||
|
<#@ assembly name="System.Core" #>
|
||||||
|
<#@ import namespace="System.Linq" #>
|
||||||
|
<#@ import namespace="System.Text" #>
|
||||||
|
<#@ import namespace="System.Collections.Generic" #>
|
||||||
|
<#@ output extension=".cs" #>
|
||||||
|
<#
|
||||||
|
var tMax = 15;
|
||||||
|
Func<int, string> typeArgs = x => string.Join(", ", Enumerable.Range(1, x).Select(x => $"T{x}")) + ", TResult";
|
||||||
|
Func<int, string> paramArgs = x => string.Join(", ", Enumerable.Range(1, x).Select(x => $"IUniTaskAsyncEnumerable<T{x}> source{x}"));
|
||||||
|
Func<int, string> parameters = x => string.Join(", ", Enumerable.Range(1, x).Select(x => $"source{x}"));
|
||||||
|
|
||||||
|
|
||||||
|
#>
|
||||||
|
using Cysharp.Threading.Tasks.Internal;
|
||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks.Linq
|
||||||
|
{
|
||||||
|
public static partial class UniTaskAsyncEnumerable
|
||||||
|
{
|
||||||
|
<# for(var i = 2; i <= tMax; i++) { #>
|
||||||
|
public static IUniTaskAsyncEnumerable<TResult> CombineLatest<<#= typeArgs(i) #>>(this <#= paramArgs(i) #>, Func<<#= typeArgs(i) #>> resultSelector)
|
||||||
|
{
|
||||||
|
<# for(var j = 1; j <= i; j++) { #>
|
||||||
|
Error.ThrowArgumentNullException(source<#= j #>, nameof(source<#= j #>));
|
||||||
|
<# } #>
|
||||||
|
Error.ThrowArgumentNullException(resultSelector, nameof(resultSelector));
|
||||||
|
|
||||||
|
return new CombineLatest<<#= typeArgs(i) #>>(<#= parameters(i) #>, resultSelector);
|
||||||
|
}
|
||||||
|
|
||||||
|
<# } #>
|
||||||
|
}
|
||||||
|
|
||||||
|
<# for(var i = 2; i <= tMax; i++) { #>
|
||||||
|
internal class CombineLatest<<#= typeArgs(i) #>> : IUniTaskAsyncEnumerable<TResult>
|
||||||
|
{
|
||||||
|
<# for(var j = 1; j <= i; j++) { #>
|
||||||
|
readonly IUniTaskAsyncEnumerable<T<#= j #>> source<#= j #>;
|
||||||
|
<# } #>
|
||||||
|
readonly Func<<#= typeArgs(i) #>> resultSelector;
|
||||||
|
|
||||||
|
public CombineLatest(<#= paramArgs(i) #>, Func<<#= typeArgs(i) #>> resultSelector)
|
||||||
|
{
|
||||||
|
<# for(var j = 1; j <= i; j++) { #>
|
||||||
|
this.source<#= j #> = source<#= j #>;
|
||||||
|
<# } #>
|
||||||
|
this.resultSelector = resultSelector;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
return new _CombineLatest(<#= parameters(i) #>, resultSelector, cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
class _CombineLatest : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
|
{
|
||||||
|
<# for(var j = 1; j <= i; j++) { #>
|
||||||
|
static readonly Action<object> Completed<#= j #>Delegate = Completed<#= j #>;
|
||||||
|
<# } #>
|
||||||
|
const int CompleteCount = <#= i #>;
|
||||||
|
|
||||||
|
<# for(var j = 1; j <= i; j++) { #>
|
||||||
|
readonly IUniTaskAsyncEnumerable<T<#= j #>> source<#= j #>;
|
||||||
|
<# } #>
|
||||||
|
readonly Func<<#= typeArgs(i) #>> resultSelector;
|
||||||
|
CancellationToken cancellationToken;
|
||||||
|
|
||||||
|
<# for(var j = 1; j <= i; j++) { #>
|
||||||
|
IUniTaskAsyncEnumerator<T<#= j #>> enumerator<#= j #>;
|
||||||
|
UniTask<bool>.Awaiter awaiter<#= j #>;
|
||||||
|
bool hasCurrent<#= j #>;
|
||||||
|
bool running<#= j #>;
|
||||||
|
T<#= j #> current<#= j #>;
|
||||||
|
|
||||||
|
<# } #>
|
||||||
|
int completedCount;
|
||||||
|
bool syncRunning;
|
||||||
|
TResult result;
|
||||||
|
|
||||||
|
public _CombineLatest(<#= paramArgs(i) #>, Func<<#= typeArgs(i) #>> resultSelector, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
<# for(var j = 1; j <= i; j++) { #>
|
||||||
|
this.source<#= j #> = source<#= j #>;
|
||||||
|
<# } #>
|
||||||
|
this.resultSelector = resultSelector;
|
||||||
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
public TResult Current => result;
|
||||||
|
|
||||||
|
public UniTask<bool> MoveNextAsync()
|
||||||
|
{
|
||||||
|
cancellationToken.ThrowIfCancellationRequested();
|
||||||
|
if (completedCount == CompleteCount) return CompletedTasks.False;
|
||||||
|
|
||||||
|
if (enumerator1 == null)
|
||||||
|
{
|
||||||
|
<# for(var j = 1; j <= i; j++) { #>
|
||||||
|
enumerator<#= j #> = source<#= j #>.GetAsyncEnumerator(cancellationToken);
|
||||||
|
<# } #>
|
||||||
|
}
|
||||||
|
|
||||||
|
completionSource.Reset();
|
||||||
|
|
||||||
|
AGAIN:
|
||||||
|
syncRunning = true;
|
||||||
|
<# for(var j = 1; j <= i; j++) { #>
|
||||||
|
if (!running<#= j #>)
|
||||||
|
{
|
||||||
|
running<#= j #> = true;
|
||||||
|
awaiter<#= j #> = enumerator<#= j #>.MoveNextAsync().GetAwaiter();
|
||||||
|
if (awaiter<#= j #>.IsCompleted)
|
||||||
|
{
|
||||||
|
Completed<#= j #>(this);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
awaiter<#= j #>.SourceOnCompleted(Completed<#= j #>Delegate, this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
<# } #>
|
||||||
|
|
||||||
|
if (<#= string.Join(" || ", Enumerable.Range(1, i).Select(x => $"!running{x}")) #>)
|
||||||
|
{
|
||||||
|
goto AGAIN;
|
||||||
|
}
|
||||||
|
syncRunning = false;
|
||||||
|
|
||||||
|
return new UniTask<bool>(this, completionSource.Version);
|
||||||
|
}
|
||||||
|
|
||||||
|
<# for(var j = 1; j <= i; j++) { #>
|
||||||
|
static void Completed<#= j #>(object state)
|
||||||
|
{
|
||||||
|
var self = (_CombineLatest)state;
|
||||||
|
self.running<#= j #> = false;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if (self.awaiter<#= j #>.GetResult())
|
||||||
|
{
|
||||||
|
self.hasCurrent<#= j #> = true;
|
||||||
|
self.current<#= j #> = self.enumerator<#= j #>.Current;
|
||||||
|
goto SUCCESS;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
self.running<#= j #> = true; // as complete, no more call MoveNextAsync.
|
||||||
|
if (Interlocked.Increment(ref self.completedCount) == CompleteCount)
|
||||||
|
{
|
||||||
|
goto COMPLETE;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
self.running<#= j #> = true; // as complete, no more call MoveNextAsync.
|
||||||
|
self.completedCount = CompleteCount;
|
||||||
|
self.completionSource.TrySetException(ex);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
SUCCESS:
|
||||||
|
if (!self.TrySetResult())
|
||||||
|
{
|
||||||
|
if (self.syncRunning) return;
|
||||||
|
self.running<#= j #> = true; // as complete, no more call MoveNextAsync.
|
||||||
|
try
|
||||||
|
{
|
||||||
|
self.awaiter<#= j #> = self.enumerator<#= j #>.MoveNextAsync().GetAwaiter();
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
self.completedCount = CompleteCount;
|
||||||
|
self.completionSource.TrySetException(ex);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.awaiter<#= j #>.SourceOnCompleted(Completed<#= j #>Delegate, self);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
COMPLETE:
|
||||||
|
self.completionSource.TrySetResult(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
<# } #>
|
||||||
|
bool TrySetResult()
|
||||||
|
{
|
||||||
|
if (<#= string.Join(" && ", Enumerable.Range(1, i).Select(x => $"hasCurrent{x}")) #>)
|
||||||
|
{
|
||||||
|
result = resultSelector(<#= string.Join(", ", Enumerable.Range(1, i).Select(x => $"current{x}")) #>);
|
||||||
|
completionSource.TrySetResult(true);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async UniTask DisposeAsync()
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
<# for(var j = 1; j <= i; j++) { #>
|
||||||
|
if (enumerator<#= j #> != null)
|
||||||
|
{
|
||||||
|
await enumerator<#= j #>.DisposeAsync();
|
||||||
|
}
|
||||||
|
<# } #>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
<# } #>
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
fileFormatVersion: 2
|
||||||
|
guid: b1b8cfa9d17af814a971ee2224aaaaa2
|
||||||
|
DefaultImporter:
|
||||||
|
externalObjects: {}
|
||||||
|
userData:
|
||||||
|
assetBundleName:
|
||||||
|
assetBundleVariant:
|
||||||
@@ -28,10 +28,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(first, second, cancellationToken);
|
return new _Concat(first, second, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
sealed class _Concat : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
||||||
{
|
{
|
||||||
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
||||||
|
|
||||||
@@ -51,12 +51,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
IUniTaskAsyncEnumerator<TSource> enumerator;
|
IUniTaskAsyncEnumerator<TSource> enumerator;
|
||||||
UniTask<bool>.Awaiter awaiter;
|
UniTask<bool>.Awaiter awaiter;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> first, IUniTaskAsyncEnumerable<TSource> second, CancellationToken cancellationToken)
|
public _Concat(IUniTaskAsyncEnumerable<TSource> first, IUniTaskAsyncEnumerable<TSource> second, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.first = first;
|
this.first = first;
|
||||||
this.second = second;
|
this.second = second;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
this.iteratingState = IteratingState.IteratingFirst;
|
this.iteratingState = IteratingState.IteratingFirst;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TSource Current { get; private set; }
|
public TSource Current { get; private set; }
|
||||||
@@ -108,7 +109,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_Concat)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -150,6 +151,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (enumerator != null)
|
if (enumerator != null)
|
||||||
{
|
{
|
||||||
return enumerator.DisposeAsync();
|
return enumerator.DisposeAsync();
|
||||||
|
|||||||
@@ -17,13 +17,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(comparer, nameof(comparer));
|
Error.ThrowArgumentNullException(comparer, nameof(comparer));
|
||||||
|
|
||||||
return Contains.InvokeAsync(source, value, comparer, cancellationToken);
|
return Contains.ContainsAsync(source, value, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static class Contains
|
internal static class Contains
|
||||||
{
|
{
|
||||||
internal static async UniTask<bool> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, TSource value, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
|
internal static async UniTask<bool> ContainsAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, TSource value, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Count.InvokeAsync(source, cancellationToken);
|
return Count.CountAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<Int32> CountAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<Int32> CountAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -18,7 +18,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Count.InvokeAsync(source, predicate, cancellationToken);
|
return Count.CountAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<Int32> CountAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<Int32> CountAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -26,7 +26,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Count.InvokeAsync(source, predicate, cancellationToken);
|
return Count.CountAwaitAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<Int32> CountAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<Int32> CountAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -34,13 +34,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Count.InvokeAsync(source, predicate, cancellationToken);
|
return Count.CountAwaitWithCancellationAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static class Count
|
internal static class Count
|
||||||
{
|
{
|
||||||
internal static async UniTask<int> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
|
internal static async UniTask<int> CountAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var count = 0;
|
var count = 0;
|
||||||
|
|
||||||
@@ -63,7 +63,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<int> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<int> CountAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var count = 0;
|
var count = 0;
|
||||||
|
|
||||||
@@ -89,7 +89,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<int> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<int> CountAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var count = 0;
|
var count = 0;
|
||||||
|
|
||||||
@@ -115,7 +115,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<int> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<int> CountAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var count = 0;
|
var count = 0;
|
||||||
|
|
||||||
|
|||||||
@@ -34,10 +34,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, defaultValue, cancellationToken);
|
return new _DefaultIfEmpty(source, defaultValue, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
sealed class _DefaultIfEmpty : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
||||||
{
|
{
|
||||||
enum IteratingState : byte
|
enum IteratingState : byte
|
||||||
{
|
{
|
||||||
@@ -56,13 +56,14 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
IUniTaskAsyncEnumerator<TSource> enumerator;
|
IUniTaskAsyncEnumerator<TSource> enumerator;
|
||||||
UniTask<bool>.Awaiter awaiter;
|
UniTask<bool>.Awaiter awaiter;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, TSource defaultValue, CancellationToken cancellationToken)
|
public _DefaultIfEmpty(IUniTaskAsyncEnumerable<TSource> source, TSource defaultValue, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.defaultValue = defaultValue;
|
this.defaultValue = defaultValue;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
|
||||||
this.iteratingState = IteratingState.Empty;
|
this.iteratingState = IteratingState.Empty;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TSource Current { get; private set; }
|
public TSource Current { get; private set; }
|
||||||
@@ -99,7 +100,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_DefaultIfEmpty)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -128,6 +129,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (enumerator != null)
|
if (enumerator != null)
|
||||||
{
|
{
|
||||||
return enumerator.DisposeAsync();
|
return enumerator.DisposeAsync();
|
||||||
|
|||||||
@@ -59,7 +59,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
|
Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
|
||||||
Error.ThrowArgumentNullException(comparer, nameof(comparer));
|
Error.ThrowArgumentNullException(comparer, nameof(comparer));
|
||||||
|
|
||||||
return new DistinctAwaitCancellation<TSource, TKey>(source, keySelector, comparer);
|
return new DistinctAwaitWithCancellation<TSource, TKey>(source, keySelector, comparer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -76,14 +76,14 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, comparer, cancellationToken);
|
return new _Distinct(source, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorBase<TSource, TSource>
|
class _Distinct : AsyncEnumeratorBase<TSource, TSource>
|
||||||
{
|
{
|
||||||
readonly HashSet<TSource> set;
|
readonly HashSet<TSource> set;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
|
public _Distinct(IUniTaskAsyncEnumerable<TSource> source, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
|
||||||
|
|
||||||
: base(source, cancellationToken)
|
: base(source, cancellationToken)
|
||||||
{
|
{
|
||||||
@@ -129,15 +129,15 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, comparer, cancellationToken);
|
return new _Distinct(source, keySelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorBase<TSource, TSource>
|
class _Distinct : AsyncEnumeratorBase<TSource, TSource>
|
||||||
{
|
{
|
||||||
readonly HashSet<TKey> set;
|
readonly HashSet<TKey> set;
|
||||||
readonly Func<TSource, TKey> keySelector;
|
readonly Func<TSource, TKey> keySelector;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _Distinct(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
|
|
||||||
: base(source, cancellationToken)
|
: base(source, cancellationToken)
|
||||||
{
|
{
|
||||||
@@ -184,15 +184,15 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, comparer, cancellationToken);
|
return new _DistinctAwait(source, keySelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorAwaitSelectorBase<TSource, TSource, TKey>
|
class _DistinctAwait : AsyncEnumeratorAwaitSelectorBase<TSource, TSource, TKey>
|
||||||
{
|
{
|
||||||
readonly HashSet<TKey> set;
|
readonly HashSet<TKey> set;
|
||||||
readonly Func<TSource, UniTask<TKey>> keySelector;
|
readonly Func<TSource, UniTask<TKey>> keySelector;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _DistinctAwait(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
|
|
||||||
: base(source, cancellationToken)
|
: base(source, cancellationToken)
|
||||||
{
|
{
|
||||||
@@ -222,13 +222,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal sealed class DistinctAwaitCancellation<TSource, TKey> : IUniTaskAsyncEnumerable<TSource>
|
internal sealed class DistinctAwaitWithCancellation<TSource, TKey> : IUniTaskAsyncEnumerable<TSource>
|
||||||
{
|
{
|
||||||
readonly IUniTaskAsyncEnumerable<TSource> source;
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
|
readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
|
||||||
readonly IEqualityComparer<TKey> comparer;
|
readonly IEqualityComparer<TKey> comparer;
|
||||||
|
|
||||||
public DistinctAwaitCancellation(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer)
|
public DistinctAwaitWithCancellation(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.keySelector = keySelector;
|
this.keySelector = keySelector;
|
||||||
@@ -237,15 +237,15 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, comparer, cancellationToken);
|
return new _DistinctAwaitWithCancellation(source, keySelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorAwaitSelectorBase<TSource, TSource, TKey>
|
class _DistinctAwaitWithCancellation : AsyncEnumeratorAwaitSelectorBase<TSource, TSource, TKey>
|
||||||
{
|
{
|
||||||
readonly HashSet<TKey> set;
|
readonly HashSet<TKey> set;
|
||||||
readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
|
readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _DistinctAwaitWithCancellation(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
|
|
||||||
: base(source, cancellationToken)
|
: base(source, cancellationToken)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -59,7 +59,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
|
Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
|
||||||
Error.ThrowArgumentNullException(comparer, nameof(comparer));
|
Error.ThrowArgumentNullException(comparer, nameof(comparer));
|
||||||
|
|
||||||
return new DistinctUntilChangedAwaitCancellation<TSource, TKey>(source, keySelector, comparer);
|
return new DistinctUntilChangedAwaitWithCancellation<TSource, TKey>(source, keySelector, comparer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -76,44 +76,126 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, comparer, cancellationToken);
|
return new _DistinctUntilChanged(source, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorBase<TSource, TSource>
|
sealed class _DistinctUntilChanged : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
||||||
{
|
{
|
||||||
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
readonly IEqualityComparer<TSource> comparer;
|
readonly IEqualityComparer<TSource> comparer;
|
||||||
TSource prev;
|
readonly CancellationToken cancellationToken;
|
||||||
bool first;
|
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
|
int state = -1;
|
||||||
|
IUniTaskAsyncEnumerator<TSource> enumerator;
|
||||||
|
UniTask<bool>.Awaiter awaiter;
|
||||||
|
Action moveNextAction;
|
||||||
|
|
||||||
: base(source, cancellationToken)
|
public _DistinctUntilChanged(IUniTaskAsyncEnumerable<TSource> source, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
|
this.source = source;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.first = true;
|
this.cancellationToken = cancellationToken;
|
||||||
|
this.moveNextAction = MoveNext;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override bool TryMoveNextCore(bool sourceHasCurrent, out bool result)
|
public TSource Current { get; private set; }
|
||||||
|
|
||||||
|
public UniTask<bool> MoveNextAsync()
|
||||||
{
|
{
|
||||||
if (sourceHasCurrent)
|
if (state == -2) return default;
|
||||||
|
|
||||||
|
completionSource.Reset();
|
||||||
|
MoveNext();
|
||||||
|
return new UniTask<bool>(this, completionSource.Version);
|
||||||
|
}
|
||||||
|
|
||||||
|
void MoveNext()
|
||||||
{
|
{
|
||||||
var v = SourceCurrent;
|
REPEAT:
|
||||||
if (first || !comparer.Equals(prev, v))
|
try
|
||||||
{
|
{
|
||||||
first = false;
|
switch (state)
|
||||||
Current = prev = v;
|
{
|
||||||
result = true;
|
case -1: // init
|
||||||
return true;
|
enumerator = source.GetAsyncEnumerator(cancellationToken);
|
||||||
|
awaiter = enumerator.MoveNextAsync().GetAwaiter();
|
||||||
|
if (awaiter.IsCompleted)
|
||||||
|
{
|
||||||
|
goto case -3;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
result = default;
|
state = -3;
|
||||||
return false;
|
awaiter.UnsafeOnCompleted(moveNextAction);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
case -3: // first
|
||||||
|
if (awaiter.GetResult())
|
||||||
|
{
|
||||||
|
Current = enumerator.Current;
|
||||||
|
goto CONTINUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
case 0: // normal
|
||||||
|
awaiter = enumerator.MoveNextAsync().GetAwaiter();
|
||||||
|
if (awaiter.IsCompleted)
|
||||||
|
{
|
||||||
|
goto case 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
state = 1;
|
||||||
|
awaiter.UnsafeOnCompleted(moveNextAction);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
case 1:
|
||||||
|
if (awaiter.GetResult())
|
||||||
|
{
|
||||||
|
var v = enumerator.Current;
|
||||||
|
if (!comparer.Equals(Current, v))
|
||||||
|
{
|
||||||
|
Current = v;
|
||||||
|
goto CONTINUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
state = 0;
|
||||||
|
goto REPEAT;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
case -2:
|
||||||
|
default:
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
state = -2;
|
||||||
|
completionSource.TrySetException(ex);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
result = false;
|
DONE:
|
||||||
return true;
|
state = -2;
|
||||||
|
completionSource.TrySetResult(false);
|
||||||
|
return;
|
||||||
|
|
||||||
|
CONTINUE:
|
||||||
|
state = 0;
|
||||||
|
completionSource.TrySetResult(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask DisposeAsync()
|
||||||
|
{
|
||||||
|
return enumerator.DisposeAsync();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -133,48 +215,131 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, comparer, cancellationToken);
|
return new _DistinctUntilChanged(source, keySelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorBase<TSource, TSource>
|
sealed class _DistinctUntilChanged : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
||||||
{
|
{
|
||||||
readonly IEqualityComparer<TKey> comparer;
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
readonly Func<TSource, TKey> keySelector;
|
readonly Func<TSource, TKey> keySelector;
|
||||||
|
readonly IEqualityComparer<TKey> comparer;
|
||||||
|
readonly CancellationToken cancellationToken;
|
||||||
|
|
||||||
|
int state = -1;
|
||||||
|
IUniTaskAsyncEnumerator<TSource> enumerator;
|
||||||
|
UniTask<bool>.Awaiter awaiter;
|
||||||
|
Action moveNextAction;
|
||||||
TKey prev;
|
TKey prev;
|
||||||
bool first;
|
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _DistinctUntilChanged(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
|
|
||||||
: base(source, cancellationToken)
|
|
||||||
{
|
{
|
||||||
this.comparer = comparer;
|
this.source = source;
|
||||||
this.keySelector = keySelector;
|
this.keySelector = keySelector;
|
||||||
this.first = true;
|
this.comparer = comparer;
|
||||||
|
this.cancellationToken = cancellationToken;
|
||||||
|
this.moveNextAction = MoveNext;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override bool TryMoveNextCore(bool sourceHasCurrent, out bool result)
|
public TSource Current { get; private set; }
|
||||||
|
|
||||||
|
public UniTask<bool> MoveNextAsync()
|
||||||
{
|
{
|
||||||
if (sourceHasCurrent)
|
if (state == -2) return default;
|
||||||
|
|
||||||
|
completionSource.Reset();
|
||||||
|
MoveNext();
|
||||||
|
return new UniTask<bool>(this, completionSource.Version);
|
||||||
|
}
|
||||||
|
|
||||||
|
void MoveNext()
|
||||||
{
|
{
|
||||||
var v = SourceCurrent;
|
REPEAT:
|
||||||
var key = keySelector(v);
|
try
|
||||||
if (first || !comparer.Equals(prev, key))
|
|
||||||
{
|
{
|
||||||
first = false;
|
switch (state)
|
||||||
prev = key;
|
{
|
||||||
Current = v;
|
case -1: // init
|
||||||
result = true;
|
enumerator = source.GetAsyncEnumerator(cancellationToken);
|
||||||
return true;
|
awaiter = enumerator.MoveNextAsync().GetAwaiter();
|
||||||
|
if (awaiter.IsCompleted)
|
||||||
|
{
|
||||||
|
goto case -3;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
result = default;
|
state = -3;
|
||||||
return false;
|
awaiter.UnsafeOnCompleted(moveNextAction);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
case -3: // first
|
||||||
|
if (awaiter.GetResult())
|
||||||
|
{
|
||||||
|
Current = enumerator.Current;
|
||||||
|
goto CONTINUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
case 0: // normal
|
||||||
|
awaiter = enumerator.MoveNextAsync().GetAwaiter();
|
||||||
|
if (awaiter.IsCompleted)
|
||||||
|
{
|
||||||
|
goto case 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
state = 1;
|
||||||
|
awaiter.UnsafeOnCompleted(moveNextAction);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
case 1:
|
||||||
|
if (awaiter.GetResult())
|
||||||
|
{
|
||||||
|
var v = enumerator.Current;
|
||||||
|
var key = keySelector(v);
|
||||||
|
if (!comparer.Equals(prev, key))
|
||||||
|
{
|
||||||
|
prev = key;
|
||||||
|
Current = v;
|
||||||
|
goto CONTINUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
state = 0;
|
||||||
|
goto REPEAT;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
case -2:
|
||||||
|
default:
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
state = -2;
|
||||||
|
completionSource.TrySetException(ex);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
result = false;
|
DONE:
|
||||||
return true;
|
state = -2;
|
||||||
|
completionSource.TrySetResult(false);
|
||||||
|
return;
|
||||||
|
|
||||||
|
CONTINUE:
|
||||||
|
state = 0;
|
||||||
|
completionSource.TrySetResult(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask DisposeAsync()
|
||||||
|
{
|
||||||
|
return enumerator.DisposeAsync();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -194,56 +359,156 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, comparer, cancellationToken);
|
return new _DistinctUntilChangedAwait(source, keySelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorAwaitSelectorBase<TSource, TSource, TKey>
|
sealed class _DistinctUntilChangedAwait : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
||||||
{
|
{
|
||||||
readonly IEqualityComparer<TKey> comparer;
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
readonly Func<TSource, UniTask<TKey>> keySelector;
|
readonly Func<TSource, UniTask<TKey>> keySelector;
|
||||||
|
readonly IEqualityComparer<TKey> comparer;
|
||||||
|
readonly CancellationToken cancellationToken;
|
||||||
|
|
||||||
|
int state = -1;
|
||||||
|
IUniTaskAsyncEnumerator<TSource> enumerator;
|
||||||
|
UniTask<bool>.Awaiter awaiter;
|
||||||
|
UniTask<TKey>.Awaiter awaiter2;
|
||||||
|
Action moveNextAction;
|
||||||
|
TSource enumeratorCurrent;
|
||||||
TKey prev;
|
TKey prev;
|
||||||
bool first;
|
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _DistinctUntilChangedAwait(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
|
|
||||||
: base(source, cancellationToken)
|
|
||||||
{
|
{
|
||||||
this.comparer = comparer;
|
this.source = source;
|
||||||
this.keySelector = keySelector;
|
this.keySelector = keySelector;
|
||||||
this.first = true;
|
this.comparer = comparer;
|
||||||
|
this.cancellationToken = cancellationToken;
|
||||||
|
this.moveNextAction = MoveNext;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override UniTask<TKey> TransformAsync(TSource sourceCurrent)
|
public TSource Current { get; private set; }
|
||||||
|
|
||||||
|
public UniTask<bool> MoveNextAsync()
|
||||||
{
|
{
|
||||||
return keySelector(sourceCurrent);
|
if (state == -2) return default;
|
||||||
|
|
||||||
|
completionSource.Reset();
|
||||||
|
MoveNext();
|
||||||
|
return new UniTask<bool>(this, completionSource.Version);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override bool TrySetCurrentCore(TKey key, out bool terminateIteration)
|
void MoveNext()
|
||||||
{
|
{
|
||||||
if (first || !comparer.Equals(prev, key))
|
REPEAT:
|
||||||
|
try
|
||||||
{
|
{
|
||||||
first = false;
|
switch (state)
|
||||||
prev = key;
|
{
|
||||||
Current = SourceCurrent;
|
case -1: // init
|
||||||
terminateIteration = false;
|
enumerator = source.GetAsyncEnumerator(cancellationToken);
|
||||||
return true;
|
awaiter = enumerator.MoveNextAsync().GetAwaiter();
|
||||||
|
if (awaiter.IsCompleted)
|
||||||
|
{
|
||||||
|
goto case -3;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
terminateIteration = false;
|
state = -3;
|
||||||
return false;
|
awaiter.UnsafeOnCompleted(moveNextAction);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
case -3: // first
|
||||||
|
if (awaiter.GetResult())
|
||||||
|
{
|
||||||
|
Current = enumerator.Current;
|
||||||
|
goto CONTINUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
case 0: // normal
|
||||||
|
awaiter = enumerator.MoveNextAsync().GetAwaiter();
|
||||||
|
if (awaiter.IsCompleted)
|
||||||
|
{
|
||||||
|
goto case 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
state = 1;
|
||||||
|
awaiter.UnsafeOnCompleted(moveNextAction);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
case 1:
|
||||||
|
if (awaiter.GetResult())
|
||||||
|
{
|
||||||
|
enumeratorCurrent = enumerator.Current;
|
||||||
|
awaiter2 = keySelector(enumeratorCurrent).GetAwaiter();
|
||||||
|
if (awaiter2.IsCompleted)
|
||||||
|
{
|
||||||
|
goto case 2;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
state = 2;
|
||||||
|
awaiter2.UnsafeOnCompleted(moveNextAction);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
case 2:
|
||||||
|
var key = awaiter2.GetResult();
|
||||||
|
if (!comparer.Equals(prev, key))
|
||||||
|
{
|
||||||
|
prev = key;
|
||||||
|
Current = enumeratorCurrent;
|
||||||
|
goto CONTINUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
state = 0;
|
||||||
|
goto REPEAT;
|
||||||
|
}
|
||||||
|
case -2:
|
||||||
|
default:
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
state = -2;
|
||||||
|
completionSource.TrySetException(ex);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
DONE:
|
||||||
|
state = -2;
|
||||||
|
completionSource.TrySetResult(false);
|
||||||
|
return;
|
||||||
|
|
||||||
|
CONTINUE:
|
||||||
|
state = 0;
|
||||||
|
completionSource.TrySetResult(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask DisposeAsync()
|
||||||
|
{
|
||||||
|
return enumerator.DisposeAsync();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal sealed class DistinctUntilChangedAwaitCancellation<TSource, TKey> : IUniTaskAsyncEnumerable<TSource>
|
internal sealed class DistinctUntilChangedAwaitWithCancellation<TSource, TKey> : IUniTaskAsyncEnumerable<TSource>
|
||||||
{
|
{
|
||||||
readonly IUniTaskAsyncEnumerable<TSource> source;
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
|
readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
|
||||||
readonly IEqualityComparer<TKey> comparer;
|
readonly IEqualityComparer<TKey> comparer;
|
||||||
|
|
||||||
public DistinctUntilChangedAwaitCancellation(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer)
|
public DistinctUntilChangedAwaitWithCancellation(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.keySelector = keySelector;
|
this.keySelector = keySelector;
|
||||||
@@ -252,45 +517,145 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, comparer, cancellationToken);
|
return new _DistinctUntilChangedAwaitWithCancellation(source, keySelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorAwaitSelectorBase<TSource, TSource, TKey>
|
sealed class _DistinctUntilChangedAwaitWithCancellation : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
||||||
{
|
{
|
||||||
readonly IEqualityComparer<TKey> comparer;
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
|
readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
|
||||||
|
readonly IEqualityComparer<TKey> comparer;
|
||||||
|
readonly CancellationToken cancellationToken;
|
||||||
|
|
||||||
|
int state = -1;
|
||||||
|
IUniTaskAsyncEnumerator<TSource> enumerator;
|
||||||
|
UniTask<bool>.Awaiter awaiter;
|
||||||
|
UniTask<TKey>.Awaiter awaiter2;
|
||||||
|
Action moveNextAction;
|
||||||
|
TSource enumeratorCurrent;
|
||||||
TKey prev;
|
TKey prev;
|
||||||
bool first;
|
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _DistinctUntilChangedAwaitWithCancellation(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
|
|
||||||
: base(source, cancellationToken)
|
|
||||||
{
|
{
|
||||||
this.comparer = comparer;
|
this.source = source;
|
||||||
this.keySelector = keySelector;
|
this.keySelector = keySelector;
|
||||||
this.first = true;
|
this.comparer = comparer;
|
||||||
|
this.cancellationToken = cancellationToken;
|
||||||
|
this.moveNextAction = MoveNext;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override UniTask<TKey> TransformAsync(TSource sourceCurrent)
|
public TSource Current { get; private set; }
|
||||||
|
|
||||||
|
public UniTask<bool> MoveNextAsync()
|
||||||
{
|
{
|
||||||
return keySelector(sourceCurrent, cancellationToken);
|
if (state == -2) return default;
|
||||||
|
|
||||||
|
completionSource.Reset();
|
||||||
|
MoveNext();
|
||||||
|
return new UniTask<bool>(this, completionSource.Version);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override bool TrySetCurrentCore(TKey key, out bool terminateIteration)
|
void MoveNext()
|
||||||
{
|
{
|
||||||
if (first || !comparer.Equals(prev, key))
|
REPEAT:
|
||||||
|
try
|
||||||
{
|
{
|
||||||
first = false;
|
switch (state)
|
||||||
prev = key;
|
{
|
||||||
Current = SourceCurrent;
|
case -1: // init
|
||||||
terminateIteration = false;
|
enumerator = source.GetAsyncEnumerator(cancellationToken);
|
||||||
return true;
|
awaiter = enumerator.MoveNextAsync().GetAwaiter();
|
||||||
|
if (awaiter.IsCompleted)
|
||||||
|
{
|
||||||
|
goto case -3;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
terminateIteration = false;
|
state = -3;
|
||||||
return false;
|
awaiter.UnsafeOnCompleted(moveNextAction);
|
||||||
}
|
return;
|
||||||
|
}
|
||||||
|
case -3: // first
|
||||||
|
if (awaiter.GetResult())
|
||||||
|
{
|
||||||
|
Current = enumerator.Current;
|
||||||
|
goto CONTINUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
case 0: // normal
|
||||||
|
awaiter = enumerator.MoveNextAsync().GetAwaiter();
|
||||||
|
if (awaiter.IsCompleted)
|
||||||
|
{
|
||||||
|
goto case 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
state = 1;
|
||||||
|
awaiter.UnsafeOnCompleted(moveNextAction);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
case 1:
|
||||||
|
if (awaiter.GetResult())
|
||||||
|
{
|
||||||
|
enumeratorCurrent = enumerator.Current;
|
||||||
|
awaiter2 = keySelector(enumeratorCurrent, cancellationToken).GetAwaiter();
|
||||||
|
if (awaiter2.IsCompleted)
|
||||||
|
{
|
||||||
|
goto case 2;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
state = 2;
|
||||||
|
awaiter2.UnsafeOnCompleted(moveNextAction);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
case 2:
|
||||||
|
var key = awaiter2.GetResult();
|
||||||
|
if (!comparer.Equals(prev, key))
|
||||||
|
{
|
||||||
|
prev = key;
|
||||||
|
Current = enumeratorCurrent;
|
||||||
|
goto CONTINUE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
state = 0;
|
||||||
|
goto REPEAT;
|
||||||
|
}
|
||||||
|
case -2:
|
||||||
|
default:
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
state = -2;
|
||||||
|
completionSource.TrySetException(ex);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
DONE:
|
||||||
|
state = -2;
|
||||||
|
completionSource.TrySetResult(false);
|
||||||
|
return;
|
||||||
|
|
||||||
|
CONTINUE:
|
||||||
|
state = 0;
|
||||||
|
completionSource.TrySetResult(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask DisposeAsync()
|
||||||
|
{
|
||||||
|
return enumerator.DisposeAsync();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -101,10 +101,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, onNext, onError, onCompleted, cancellationToken);
|
return new _Do(source, onNext, onError, onCompleted, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
sealed class _Do : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
|
||||||
{
|
{
|
||||||
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
||||||
|
|
||||||
@@ -117,13 +117,14 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
IUniTaskAsyncEnumerator<TSource> enumerator;
|
IUniTaskAsyncEnumerator<TSource> enumerator;
|
||||||
UniTask<bool>.Awaiter awaiter;
|
UniTask<bool>.Awaiter awaiter;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Action<TSource> onNext, Action<Exception> onError, Action onCompleted, CancellationToken cancellationToken)
|
public _Do(IUniTaskAsyncEnumerable<TSource> source, Action<TSource> onNext, Action<Exception> onError, Action onCompleted, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.onNext = onNext;
|
this.onNext = onNext;
|
||||||
this.onError = onError;
|
this.onError = onError;
|
||||||
this.onCompleted = onCompleted;
|
this.onCompleted = onCompleted;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TSource Current { get; private set; }
|
public TSource Current { get; private set; }
|
||||||
@@ -199,7 +200,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_Do)state;
|
||||||
|
|
||||||
if (self.TryGetResultWithNotification(self.awaiter, out var result))
|
if (self.TryGetResultWithNotification(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -244,6 +245,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (enumerator != null)
|
if (enumerator != null)
|
||||||
{
|
{
|
||||||
return enumerator.DisposeAsync();
|
return enumerator.DisposeAsync();
|
||||||
|
|||||||
@@ -10,20 +10,20 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return ElementAt.InvokeAsync(source, index, cancellationToken, false);
|
return ElementAt.ElementAtAsync(source, index, cancellationToken, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> ElementAtOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, int index, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> ElementAtOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, int index, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return ElementAt.InvokeAsync(source, index, cancellationToken, true);
|
return ElementAt.ElementAtAsync(source, index, cancellationToken, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static class ElementAt
|
internal static class ElementAt
|
||||||
{
|
{
|
||||||
public static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, int index, CancellationToken cancellationToken, bool defaultIfEmpty)
|
public static async UniTask<TSource> ElementAtAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, int index, CancellationToken cancellationToken, bool defaultIfEmpty)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
|
|||||||
@@ -20,14 +20,14 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return Enumerator.Instance;
|
return _Empty.Instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : IUniTaskAsyncEnumerator<T>
|
class _Empty : IUniTaskAsyncEnumerator<T>
|
||||||
{
|
{
|
||||||
public static readonly IUniTaskAsyncEnumerator<T> Instance = new Enumerator();
|
public static readonly IUniTaskAsyncEnumerator<T> Instance = new _Empty();
|
||||||
|
|
||||||
Enumerator()
|
_Empty()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -40,10 +40,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(first, second, comparer, cancellationToken);
|
return new _Except(first, second, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorBase<TSource, TSource>
|
class _Except : AsyncEnumeratorBase<TSource, TSource>
|
||||||
{
|
{
|
||||||
static Action<object> HashSetAsyncCoreDelegate = HashSetAsyncCore;
|
static Action<object> HashSetAsyncCoreDelegate = HashSetAsyncCore;
|
||||||
|
|
||||||
@@ -53,7 +53,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
HashSet<TSource> set;
|
HashSet<TSource> set;
|
||||||
UniTask<HashSet<TSource>>.Awaiter awaiter;
|
UniTask<HashSet<TSource>>.Awaiter awaiter;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> first, IUniTaskAsyncEnumerable<TSource> second, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
|
public _Except(IUniTaskAsyncEnumerable<TSource> first, IUniTaskAsyncEnumerable<TSource> second, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
|
||||||
|
|
||||||
: base(first, cancellationToken)
|
: base(first, cancellationToken)
|
||||||
{
|
{
|
||||||
@@ -81,7 +81,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void HashSetAsyncCore(object state)
|
static void HashSetAsyncCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_Except)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return First.InvokeAsync(source, cancellationToken, false);
|
return First.FirstAsync(source, cancellationToken, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> FirstAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> FirstAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -18,7 +18,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return First.InvokeAsync(source, predicate, cancellationToken, false);
|
return First.FirstAsync(source, predicate, cancellationToken, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> FirstAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> FirstAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -26,7 +26,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return First.InvokeAsync(source, predicate, cancellationToken, false);
|
return First.FirstAwaitAsync(source, predicate, cancellationToken, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> FirstAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> FirstAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -34,14 +34,14 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return First.InvokeAsync(source, predicate, cancellationToken, false);
|
return First.FirstAwaitWithCancellationAsync(source, predicate, cancellationToken, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> FirstOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> FirstOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return First.InvokeAsync(source, cancellationToken, true);
|
return First.FirstAsync(source, cancellationToken, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> FirstOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> FirstOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -49,7 +49,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return First.InvokeAsync(source, predicate, cancellationToken, true);
|
return First.FirstAsync(source, predicate, cancellationToken, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> FirstOrDefaultAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> FirstOrDefaultAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -57,7 +57,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return First.InvokeAsync(source, predicate, cancellationToken, true);
|
return First.FirstAwaitAsync(source, predicate, cancellationToken, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> FirstOrDefaultAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> FirstOrDefaultAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -65,13 +65,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return First.InvokeAsync(source, predicate, cancellationToken, true);
|
return First.FirstAwaitWithCancellationAsync(source, predicate, cancellationToken, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static class First
|
internal static class First
|
||||||
{
|
{
|
||||||
public static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken, bool defaultIfEmpty)
|
public static async UniTask<TSource> FirstAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken, bool defaultIfEmpty)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -101,7 +101,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
public static async UniTask<TSource> FirstAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -133,7 +133,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
public static async UniTask<TSource> FirstAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -165,7 +165,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
public static async UniTask<TSource> FirstAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(action, nameof(action));
|
Error.ThrowArgumentNullException(action, nameof(action));
|
||||||
|
|
||||||
return Cysharp.Threading.Tasks.Linq.ForEach.InvokeAsync(source, action, cancellationToken);
|
return Cysharp.Threading.Tasks.Linq.ForEach.ForEachAsync(source, action, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask ForEachAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Action<TSource, Int32> action, CancellationToken cancellationToken = default)
|
public static UniTask ForEachAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Action<TSource, Int32> action, CancellationToken cancellationToken = default)
|
||||||
@@ -19,7 +19,23 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(action, nameof(action));
|
Error.ThrowArgumentNullException(action, nameof(action));
|
||||||
|
|
||||||
return Cysharp.Threading.Tasks.Linq.ForEach.InvokeAsync(source, action, cancellationToken);
|
return Cysharp.Threading.Tasks.Linq.ForEach.ForEachAsync(source, action, cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Obsolete(Error), Use Use ForEachAwaitAsync instead.</summary>
|
||||||
|
[Obsolete("Use ForEachAwaitAsync instead.", true)]
|
||||||
|
[System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)]
|
||||||
|
public static UniTask ForEachAsync<T>(this IUniTaskAsyncEnumerable<T> source, Func<T, UniTask> action, CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
throw new NotSupportedException("Use ForEachAwaitAsync instead.");
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Obsolete(Error), Use Use ForEachAwaitAsync instead.</summary>
|
||||||
|
[Obsolete("Use ForEachAwaitAsync instead.", true)]
|
||||||
|
[System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)]
|
||||||
|
public static UniTask ForEachAsync<T>(this IUniTaskAsyncEnumerable<T> source, Func<T, int, UniTask> action, CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
throw new NotSupportedException("Use ForEachAwaitAsync instead.");
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask ForEachAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask> action, CancellationToken cancellationToken = default)
|
public static UniTask ForEachAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask> action, CancellationToken cancellationToken = default)
|
||||||
@@ -27,7 +43,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(action, nameof(action));
|
Error.ThrowArgumentNullException(action, nameof(action));
|
||||||
|
|
||||||
return Cysharp.Threading.Tasks.Linq.ForEach.InvokeAwaitAsync(source, action, cancellationToken);
|
return Cysharp.Threading.Tasks.Linq.ForEach.ForEachAwaitAsync(source, action, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask ForEachAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32, UniTask> action, CancellationToken cancellationToken = default)
|
public static UniTask ForEachAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32, UniTask> action, CancellationToken cancellationToken = default)
|
||||||
@@ -35,7 +51,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(action, nameof(action));
|
Error.ThrowArgumentNullException(action, nameof(action));
|
||||||
|
|
||||||
return Cysharp.Threading.Tasks.Linq.ForEach.InvokeAwaitAsync(source, action, cancellationToken);
|
return Cysharp.Threading.Tasks.Linq.ForEach.ForEachAwaitAsync(source, action, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask ForEachAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask> action, CancellationToken cancellationToken = default)
|
public static UniTask ForEachAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask> action, CancellationToken cancellationToken = default)
|
||||||
@@ -43,7 +59,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(action, nameof(action));
|
Error.ThrowArgumentNullException(action, nameof(action));
|
||||||
|
|
||||||
return Cysharp.Threading.Tasks.Linq.ForEach.InvokeAwaitWithCancellationAsync(source, action, cancellationToken);
|
return Cysharp.Threading.Tasks.Linq.ForEach.ForEachAwaitWithCancellationAsync(source, action, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask ForEachAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32, CancellationToken, UniTask> action, CancellationToken cancellationToken = default)
|
public static UniTask ForEachAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32, CancellationToken, UniTask> action, CancellationToken cancellationToken = default)
|
||||||
@@ -51,13 +67,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(action, nameof(action));
|
Error.ThrowArgumentNullException(action, nameof(action));
|
||||||
|
|
||||||
return Cysharp.Threading.Tasks.Linq.ForEach.InvokeAwaitWithCancellationAsync(source, action, cancellationToken);
|
return Cysharp.Threading.Tasks.Linq.ForEach.ForEachAwaitWithCancellationAsync(source, action, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static class ForEach
|
internal static class ForEach
|
||||||
{
|
{
|
||||||
public static async UniTask InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Action<TSource> action, CancellationToken cancellationToken)
|
public static async UniTask ForEachAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Action<TSource> action, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -76,7 +92,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Action<TSource, Int32> action, CancellationToken cancellationToken)
|
public static async UniTask ForEachAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Action<TSource, Int32> action, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -96,7 +112,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask InvokeAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask> action, CancellationToken cancellationToken)
|
public static async UniTask ForEachAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask> action, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -115,7 +131,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask InvokeAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32, UniTask> action, CancellationToken cancellationToken)
|
public static async UniTask ForEachAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32, UniTask> action, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -135,7 +151,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask InvokeAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask> action, CancellationToken cancellationToken)
|
public static async UniTask ForEachAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask> action, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -154,7 +170,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask InvokeAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32, CancellationToken, UniTask> action, CancellationToken cancellationToken)
|
public static async UniTask ForEachAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Int32, CancellationToken, UniTask> action, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
|
|||||||
@@ -235,10 +235,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, elementSelector, comparer, cancellationToken);
|
return new _GroupBy(source, keySelector, elementSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>>
|
sealed class _GroupBy : MoveNextSource, IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>>
|
||||||
{
|
{
|
||||||
readonly IUniTaskAsyncEnumerable<TSource> source;
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
readonly Func<TSource, TKey> keySelector;
|
readonly Func<TSource, TKey> keySelector;
|
||||||
@@ -248,13 +248,14 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _GroupBy(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.keySelector = keySelector;
|
this.keySelector = keySelector;
|
||||||
this.elementSelector = elementSelector;
|
this.elementSelector = elementSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public IGrouping<TKey, TElement> Current { get; private set; }
|
public IGrouping<TKey, TElement> Current { get; private set; }
|
||||||
@@ -313,6 +314,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (groupEnumerator != null)
|
if (groupEnumerator != null)
|
||||||
{
|
{
|
||||||
groupEnumerator.Dispose();
|
groupEnumerator.Dispose();
|
||||||
@@ -342,10 +344,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, elementSelector, resultSelector, comparer, cancellationToken);
|
return new _GroupBy(source, keySelector, elementSelector, resultSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
sealed class _GroupBy : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
{
|
{
|
||||||
readonly IUniTaskAsyncEnumerable<TSource> source;
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
readonly Func<TSource, TKey> keySelector;
|
readonly Func<TSource, TKey> keySelector;
|
||||||
@@ -356,7 +358,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, Func<TKey, IEnumerable<TElement>, TResult> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _GroupBy(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, Func<TKey, IEnumerable<TElement>, TResult> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.keySelector = keySelector;
|
this.keySelector = keySelector;
|
||||||
@@ -364,6 +366,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
this.resultSelector = resultSelector;
|
this.resultSelector = resultSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TResult Current { get; private set; }
|
public TResult Current { get; private set; }
|
||||||
@@ -423,6 +426,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (groupEnumerator != null)
|
if (groupEnumerator != null)
|
||||||
{
|
{
|
||||||
groupEnumerator.Dispose();
|
groupEnumerator.Dispose();
|
||||||
@@ -450,10 +454,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, elementSelector, comparer, cancellationToken);
|
return new _GroupByAwait(source, keySelector, elementSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>>
|
sealed class _GroupByAwait : MoveNextSource, IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>>
|
||||||
{
|
{
|
||||||
readonly IUniTaskAsyncEnumerable<TSource> source;
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
readonly Func<TSource, UniTask<TKey>> keySelector;
|
readonly Func<TSource, UniTask<TKey>> keySelector;
|
||||||
@@ -463,13 +467,14 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, Func<TSource, UniTask<TElement>> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _GroupByAwait(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, Func<TSource, UniTask<TElement>> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.keySelector = keySelector;
|
this.keySelector = keySelector;
|
||||||
this.elementSelector = elementSelector;
|
this.elementSelector = elementSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public IGrouping<TKey, TElement> Current { get; private set; }
|
public IGrouping<TKey, TElement> Current { get; private set; }
|
||||||
@@ -528,6 +533,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (groupEnumerator != null)
|
if (groupEnumerator != null)
|
||||||
{
|
{
|
||||||
groupEnumerator.Dispose();
|
groupEnumerator.Dispose();
|
||||||
@@ -557,10 +563,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, elementSelector, resultSelector, comparer, cancellationToken);
|
return new _GroupByAwait(source, keySelector, elementSelector, resultSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
sealed class _GroupByAwait : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
{
|
{
|
||||||
readonly static Action<object> ResultSelectCoreDelegate = ResultSelectCore;
|
readonly static Action<object> ResultSelectCoreDelegate = ResultSelectCore;
|
||||||
|
|
||||||
@@ -574,7 +580,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
||||||
UniTask<TResult>.Awaiter awaiter;
|
UniTask<TResult>.Awaiter awaiter;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, Func<TSource, UniTask<TElement>> elementSelector, Func<TKey, IEnumerable<TElement>, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _GroupByAwait(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, Func<TSource, UniTask<TElement>> elementSelector, Func<TKey, IEnumerable<TElement>, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.keySelector = keySelector;
|
this.keySelector = keySelector;
|
||||||
@@ -582,6 +588,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
this.resultSelector = resultSelector;
|
this.resultSelector = resultSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TResult Current { get; private set; }
|
public TResult Current { get; private set; }
|
||||||
@@ -650,7 +657,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void ResultSelectCore(object state)
|
static void ResultSelectCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_GroupByAwait)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -661,6 +668,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (groupEnumerator != null)
|
if (groupEnumerator != null)
|
||||||
{
|
{
|
||||||
groupEnumerator.Dispose();
|
groupEnumerator.Dispose();
|
||||||
@@ -688,10 +696,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, elementSelector, comparer, cancellationToken);
|
return new _GroupByAwaitWithCancellation(source, keySelector, elementSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>>
|
sealed class _GroupByAwaitWithCancellation : MoveNextSource, IUniTaskAsyncEnumerator<IGrouping<TKey, TElement>>
|
||||||
{
|
{
|
||||||
readonly IUniTaskAsyncEnumerable<TSource> source;
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
|
readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
|
||||||
@@ -701,13 +709,14 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, Func<TSource, CancellationToken, UniTask<TElement>> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _GroupByAwaitWithCancellation(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, Func<TSource, CancellationToken, UniTask<TElement>> elementSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.keySelector = keySelector;
|
this.keySelector = keySelector;
|
||||||
this.elementSelector = elementSelector;
|
this.elementSelector = elementSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public IGrouping<TKey, TElement> Current { get; private set; }
|
public IGrouping<TKey, TElement> Current { get; private set; }
|
||||||
@@ -766,6 +775,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (groupEnumerator != null)
|
if (groupEnumerator != null)
|
||||||
{
|
{
|
||||||
groupEnumerator.Dispose();
|
groupEnumerator.Dispose();
|
||||||
@@ -795,10 +805,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, keySelector, elementSelector, resultSelector, comparer, cancellationToken);
|
return new _GroupByAwaitWithCancellation(source, keySelector, elementSelector, resultSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
sealed class _GroupByAwaitWithCancellation : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
{
|
{
|
||||||
readonly static Action<object> ResultSelectCoreDelegate = ResultSelectCore;
|
readonly static Action<object> ResultSelectCoreDelegate = ResultSelectCore;
|
||||||
|
|
||||||
@@ -812,7 +822,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
IEnumerator<IGrouping<TKey, TElement>> groupEnumerator;
|
||||||
UniTask<TResult>.Awaiter awaiter;
|
UniTask<TResult>.Awaiter awaiter;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, Func<TSource, CancellationToken, UniTask<TElement>> elementSelector, Func<TKey, IEnumerable<TElement>, CancellationToken, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _GroupByAwaitWithCancellation(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, Func<TSource, CancellationToken, UniTask<TElement>> elementSelector, Func<TKey, IEnumerable<TElement>, CancellationToken, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.source = source;
|
this.source = source;
|
||||||
this.keySelector = keySelector;
|
this.keySelector = keySelector;
|
||||||
@@ -820,6 +830,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
this.resultSelector = resultSelector;
|
this.resultSelector = resultSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TResult Current { get; private set; }
|
public TResult Current { get; private set; }
|
||||||
@@ -888,7 +899,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void ResultSelectCore(object state)
|
static void ResultSelectCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_GroupByAwaitWithCancellation)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -899,6 +910,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (groupEnumerator != null)
|
if (groupEnumerator != null)
|
||||||
{
|
{
|
||||||
groupEnumerator.Dispose();
|
groupEnumerator.Dispose();
|
||||||
|
|||||||
@@ -62,7 +62,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(innerKeySelector, nameof(innerKeySelector));
|
Error.ThrowArgumentNullException(innerKeySelector, nameof(innerKeySelector));
|
||||||
Error.ThrowArgumentNullException(resultSelector, nameof(resultSelector));
|
Error.ThrowArgumentNullException(resultSelector, nameof(resultSelector));
|
||||||
|
|
||||||
return new GroupJoinWithCancellationAwait<TOuter, TInner, TKey, TResult>(outer, inner, outerKeySelector, innerKeySelector, resultSelector, EqualityComparer<TKey>.Default);
|
return new GroupJoinAwaitWithCancellation<TOuter, TInner, TKey, TResult>(outer, inner, outerKeySelector, innerKeySelector, resultSelector, EqualityComparer<TKey>.Default);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static IUniTaskAsyncEnumerable<TResult> GroupJoinAwaitWithCancellation<TOuter, TInner, TKey, TResult>(this IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, CancellationToken, UniTask<TKey>> outerKeySelector, Func<TInner, CancellationToken, UniTask<TKey>> innerKeySelector, Func<TOuter, IEnumerable<TInner>, CancellationToken, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer)
|
public static IUniTaskAsyncEnumerable<TResult> GroupJoinAwaitWithCancellation<TOuter, TInner, TKey, TResult>(this IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, CancellationToken, UniTask<TKey>> outerKeySelector, Func<TInner, CancellationToken, UniTask<TKey>> innerKeySelector, Func<TOuter, IEnumerable<TInner>, CancellationToken, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer)
|
||||||
@@ -74,7 +74,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(resultSelector, nameof(resultSelector));
|
Error.ThrowArgumentNullException(resultSelector, nameof(resultSelector));
|
||||||
Error.ThrowArgumentNullException(comparer, nameof(comparer));
|
Error.ThrowArgumentNullException(comparer, nameof(comparer));
|
||||||
|
|
||||||
return new GroupJoinWithCancellationAwait<TOuter, TInner, TKey, TResult>(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer);
|
return new GroupJoinAwaitWithCancellation<TOuter, TInner, TKey, TResult>(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -100,10 +100,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
return new _GroupJoin(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
sealed class _GroupJoin : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
{
|
{
|
||||||
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
||||||
|
|
||||||
@@ -120,7 +120,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
UniTask<bool>.Awaiter awaiter;
|
UniTask<bool>.Awaiter awaiter;
|
||||||
|
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, TKey> outerKeySelector, Func<TInner, TKey> innerKeySelector, Func<TOuter, IEnumerable<TInner>, TResult> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _GroupJoin(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, TKey> outerKeySelector, Func<TInner, TKey> innerKeySelector, Func<TOuter, IEnumerable<TInner>, TResult> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.outer = outer;
|
this.outer = outer;
|
||||||
this.inner = inner;
|
this.inner = inner;
|
||||||
@@ -129,6 +129,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
this.resultSelector = resultSelector;
|
this.resultSelector = resultSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TResult Current { get; private set; }
|
public TResult Current { get; private set; }
|
||||||
@@ -186,7 +187,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_GroupJoin)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -208,6 +209,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (enumerator != null)
|
if (enumerator != null)
|
||||||
{
|
{
|
||||||
return enumerator.DisposeAsync();
|
return enumerator.DisposeAsync();
|
||||||
@@ -239,10 +241,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
return new _GroupJoinAwait(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
sealed class _GroupJoinAwait : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
{
|
{
|
||||||
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
||||||
readonly static Action<object> ResultSelectCoreDelegate = ResultSelectCore;
|
readonly static Action<object> ResultSelectCoreDelegate = ResultSelectCore;
|
||||||
@@ -264,7 +266,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
UniTask<TResult>.Awaiter resultAwaiter;
|
UniTask<TResult>.Awaiter resultAwaiter;
|
||||||
|
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, UniTask<TKey>> outerKeySelector, Func<TInner, UniTask<TKey>> innerKeySelector, Func<TOuter, IEnumerable<TInner>, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _GroupJoinAwait(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, UniTask<TKey>> outerKeySelector, Func<TInner, UniTask<TKey>> innerKeySelector, Func<TOuter, IEnumerable<TInner>, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.outer = outer;
|
this.outer = outer;
|
||||||
this.inner = inner;
|
this.inner = inner;
|
||||||
@@ -273,6 +275,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
this.resultSelector = resultSelector;
|
this.resultSelector = resultSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TResult Current { get; private set; }
|
public TResult Current { get; private set; }
|
||||||
@@ -330,7 +333,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_GroupJoinAwait)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -364,7 +367,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void OuterKeySelectCore(object state)
|
static void OuterKeySelectCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_GroupJoinAwait)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.outerKeyAwaiter, out var result))
|
if (self.TryGetResult(self.outerKeyAwaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -390,7 +393,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void ResultSelectCore(object state)
|
static void ResultSelectCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_GroupJoinAwait)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.resultAwaiter, out var result))
|
if (self.TryGetResult(self.resultAwaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -401,6 +404,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (enumerator != null)
|
if (enumerator != null)
|
||||||
{
|
{
|
||||||
return enumerator.DisposeAsync();
|
return enumerator.DisposeAsync();
|
||||||
@@ -411,7 +415,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal sealed class GroupJoinWithCancellationAwait<TOuter, TInner, TKey, TResult> : IUniTaskAsyncEnumerable<TResult>
|
internal sealed class GroupJoinAwaitWithCancellation<TOuter, TInner, TKey, TResult> : IUniTaskAsyncEnumerable<TResult>
|
||||||
{
|
{
|
||||||
readonly IUniTaskAsyncEnumerable<TOuter> outer;
|
readonly IUniTaskAsyncEnumerable<TOuter> outer;
|
||||||
readonly IUniTaskAsyncEnumerable<TInner> inner;
|
readonly IUniTaskAsyncEnumerable<TInner> inner;
|
||||||
@@ -420,7 +424,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
readonly Func<TOuter, IEnumerable<TInner>, CancellationToken, UniTask<TResult>> resultSelector;
|
readonly Func<TOuter, IEnumerable<TInner>, CancellationToken, UniTask<TResult>> resultSelector;
|
||||||
readonly IEqualityComparer<TKey> comparer;
|
readonly IEqualityComparer<TKey> comparer;
|
||||||
|
|
||||||
public GroupJoinWithCancellationAwait(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, CancellationToken, UniTask<TKey>> outerKeySelector, Func<TInner, CancellationToken, UniTask<TKey>> innerKeySelector, Func<TOuter, IEnumerable<TInner>, CancellationToken, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer)
|
public GroupJoinAwaitWithCancellation(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, CancellationToken, UniTask<TKey>> outerKeySelector, Func<TInner, CancellationToken, UniTask<TKey>> innerKeySelector, Func<TOuter, IEnumerable<TInner>, CancellationToken, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer)
|
||||||
{
|
{
|
||||||
this.outer = outer;
|
this.outer = outer;
|
||||||
this.inner = inner;
|
this.inner = inner;
|
||||||
@@ -432,10 +436,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
return new _GroupJoinAwaitWithCancellation(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
sealed class _GroupJoinAwaitWithCancellation : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
{
|
{
|
||||||
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
||||||
readonly static Action<object> ResultSelectCoreDelegate = ResultSelectCore;
|
readonly static Action<object> ResultSelectCoreDelegate = ResultSelectCore;
|
||||||
@@ -457,7 +461,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
UniTask<TResult>.Awaiter resultAwaiter;
|
UniTask<TResult>.Awaiter resultAwaiter;
|
||||||
|
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, CancellationToken, UniTask<TKey>> outerKeySelector, Func<TInner, CancellationToken, UniTask<TKey>> innerKeySelector, Func<TOuter, IEnumerable<TInner>, CancellationToken, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _GroupJoinAwaitWithCancellation(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, CancellationToken, UniTask<TKey>> outerKeySelector, Func<TInner, CancellationToken, UniTask<TKey>> innerKeySelector, Func<TOuter, IEnumerable<TInner>, CancellationToken, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.outer = outer;
|
this.outer = outer;
|
||||||
this.inner = inner;
|
this.inner = inner;
|
||||||
@@ -466,6 +470,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
this.resultSelector = resultSelector;
|
this.resultSelector = resultSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TResult Current { get; private set; }
|
public TResult Current { get; private set; }
|
||||||
@@ -523,7 +528,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_GroupJoinAwaitWithCancellation)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -557,7 +562,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void OuterKeySelectCore(object state)
|
static void OuterKeySelectCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_GroupJoinAwaitWithCancellation)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.outerKeyAwaiter, out var result))
|
if (self.TryGetResult(self.outerKeyAwaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -583,7 +588,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void ResultSelectCore(object state)
|
static void ResultSelectCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_GroupJoinAwaitWithCancellation)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.resultAwaiter, out var result))
|
if (self.TryGetResult(self.resultAwaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -594,6 +599,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (enumerator != null)
|
if (enumerator != null)
|
||||||
{
|
{
|
||||||
return enumerator.DisposeAsync();
|
return enumerator.DisposeAsync();
|
||||||
|
|||||||
@@ -40,10 +40,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(first, second, comparer, cancellationToken);
|
return new _Intersect(first, second, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorBase<TSource, TSource>
|
class _Intersect : AsyncEnumeratorBase<TSource, TSource>
|
||||||
{
|
{
|
||||||
static Action<object> HashSetAsyncCoreDelegate = HashSetAsyncCore;
|
static Action<object> HashSetAsyncCoreDelegate = HashSetAsyncCore;
|
||||||
|
|
||||||
@@ -53,7 +53,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
HashSet<TSource> set;
|
HashSet<TSource> set;
|
||||||
UniTask<HashSet<TSource>>.Awaiter awaiter;
|
UniTask<HashSet<TSource>>.Awaiter awaiter;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TSource> first, IUniTaskAsyncEnumerable<TSource> second, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
|
public _Intersect(IUniTaskAsyncEnumerable<TSource> first, IUniTaskAsyncEnumerable<TSource> second, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
|
||||||
|
|
||||||
: base(first, cancellationToken)
|
: base(first, cancellationToken)
|
||||||
{
|
{
|
||||||
@@ -81,7 +81,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void HashSetAsyncCore(object state)
|
static void HashSetAsyncCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_Intersect)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -99,10 +99,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
return new _Join(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
sealed class _Join : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
{
|
{
|
||||||
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
||||||
|
|
||||||
@@ -122,7 +122,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
bool continueNext;
|
bool continueNext;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, TKey> outerKeySelector, Func<TInner, TKey> innerKeySelector, Func<TOuter, TInner, TResult> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _Join(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, TKey> outerKeySelector, Func<TInner, TKey> innerKeySelector, Func<TOuter, TInner, TResult> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.outer = outer;
|
this.outer = outer;
|
||||||
this.inner = inner;
|
this.inner = inner;
|
||||||
@@ -131,6 +131,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
this.resultSelector = resultSelector;
|
this.resultSelector = resultSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TResult Current { get; private set; }
|
public TResult Current { get; private set; }
|
||||||
@@ -215,7 +216,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_Join)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -248,6 +249,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (valueEnumerator != null)
|
if (valueEnumerator != null)
|
||||||
{
|
{
|
||||||
valueEnumerator.Dispose();
|
valueEnumerator.Dispose();
|
||||||
@@ -284,10 +286,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
return new _JoinAwait(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
sealed class _JoinAwait : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
{
|
{
|
||||||
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
||||||
static readonly Action<object> OuterSelectCoreDelegate = OuterSelectCore;
|
static readonly Action<object> OuterSelectCoreDelegate = OuterSelectCore;
|
||||||
@@ -312,7 +314,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
bool continueNext;
|
bool continueNext;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, UniTask<TKey>> outerKeySelector, Func<TInner, UniTask<TKey>> innerKeySelector, Func<TOuter, TInner, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _JoinAwait(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, UniTask<TKey>> outerKeySelector, Func<TInner, UniTask<TKey>> innerKeySelector, Func<TOuter, TInner, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.outer = outer;
|
this.outer = outer;
|
||||||
this.inner = inner;
|
this.inner = inner;
|
||||||
@@ -321,6 +323,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
this.resultSelector = resultSelector;
|
this.resultSelector = resultSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TResult Current { get; private set; }
|
public TResult Current { get; private set; }
|
||||||
@@ -408,7 +411,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_JoinAwait)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -442,7 +445,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void OuterSelectCore(object state)
|
static void OuterSelectCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_JoinAwait)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.outerKeyAwaiter, out var key))
|
if (self.TryGetResult(self.outerKeyAwaiter, out var key))
|
||||||
{
|
{
|
||||||
@@ -465,7 +468,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void ResultSelectCore(object state)
|
static void ResultSelectCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_JoinAwait)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.resultAwaiter, out var result))
|
if (self.TryGetResult(self.resultAwaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -476,6 +479,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (valueEnumerator != null)
|
if (valueEnumerator != null)
|
||||||
{
|
{
|
||||||
valueEnumerator.Dispose();
|
valueEnumerator.Dispose();
|
||||||
@@ -512,10 +516,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
return new _JoinAwaitWithCancellation(outer, inner, outerKeySelector, innerKeySelector, resultSelector, comparer, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
sealed class _JoinAwaitWithCancellation : MoveNextSource, IUniTaskAsyncEnumerator<TResult>
|
||||||
{
|
{
|
||||||
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
||||||
static readonly Action<object> OuterSelectCoreDelegate = OuterSelectCore;
|
static readonly Action<object> OuterSelectCoreDelegate = OuterSelectCore;
|
||||||
@@ -540,7 +544,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
bool continueNext;
|
bool continueNext;
|
||||||
|
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, CancellationToken, UniTask<TKey>> outerKeySelector, Func<TInner, CancellationToken, UniTask<TKey>> innerKeySelector, Func<TOuter, TInner, CancellationToken, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
public _JoinAwaitWithCancellation(IUniTaskAsyncEnumerable<TOuter> outer, IUniTaskAsyncEnumerable<TInner> inner, Func<TOuter, CancellationToken, UniTask<TKey>> outerKeySelector, Func<TInner, CancellationToken, UniTask<TKey>> innerKeySelector, Func<TOuter, TInner, CancellationToken, UniTask<TResult>> resultSelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.outer = outer;
|
this.outer = outer;
|
||||||
this.inner = inner;
|
this.inner = inner;
|
||||||
@@ -549,6 +553,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
this.resultSelector = resultSelector;
|
this.resultSelector = resultSelector;
|
||||||
this.comparer = comparer;
|
this.comparer = comparer;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TResult Current { get; private set; }
|
public TResult Current { get; private set; }
|
||||||
@@ -636,7 +641,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void MoveNextCore(object state)
|
static void MoveNextCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_JoinAwaitWithCancellation)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.awaiter, out var result))
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -670,7 +675,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void OuterSelectCore(object state)
|
static void OuterSelectCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_JoinAwaitWithCancellation)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.outerKeyAwaiter, out var key))
|
if (self.TryGetResult(self.outerKeyAwaiter, out var key))
|
||||||
{
|
{
|
||||||
@@ -693,7 +698,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
static void ResultSelectCore(object state)
|
static void ResultSelectCore(object state)
|
||||||
{
|
{
|
||||||
var self = (Enumerator)state;
|
var self = (_JoinAwaitWithCancellation)state;
|
||||||
|
|
||||||
if (self.TryGetResult(self.resultAwaiter, out var result))
|
if (self.TryGetResult(self.resultAwaiter, out var result))
|
||||||
{
|
{
|
||||||
@@ -704,6 +709,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
if (valueEnumerator != null)
|
if (valueEnumerator != null)
|
||||||
{
|
{
|
||||||
valueEnumerator.Dispose();
|
valueEnumerator.Dispose();
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Last.InvokeAsync(source, cancellationToken, false);
|
return Last.LastAsync(source, cancellationToken, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> LastAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> LastAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -18,7 +18,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Last.InvokeAsync(source, predicate, cancellationToken, false);
|
return Last.LastAsync(source, predicate, cancellationToken, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> LastAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> LastAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -26,7 +26,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Last.InvokeAsync(source, predicate, cancellationToken, false);
|
return Last.LastAwaitAsync(source, predicate, cancellationToken, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> LastAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> LastAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -34,14 +34,14 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Last.InvokeAsync(source, predicate, cancellationToken, false);
|
return Last.LastAwaitWithCancellationAsync(source, predicate, cancellationToken, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> LastOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> LastOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Last.InvokeAsync(source, cancellationToken, true);
|
return Last.LastAsync(source, cancellationToken, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> LastOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> LastOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -49,7 +49,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Last.InvokeAsync(source, predicate, cancellationToken, true);
|
return Last.LastAsync(source, predicate, cancellationToken, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> LastOrDefaultAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> LastOrDefaultAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -57,7 +57,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Last.InvokeAsync(source, predicate, cancellationToken, true);
|
return Last.LastAwaitAsync(source, predicate, cancellationToken, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TSource> LastOrDefaultAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<TSource> LastOrDefaultAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -65,13 +65,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return Last.InvokeAsync(source, predicate, cancellationToken, true);
|
return Last.LastAwaitWithCancellationAsync(source, predicate, cancellationToken, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static class Last
|
internal static class Last
|
||||||
{
|
{
|
||||||
public static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken, bool defaultIfEmpty)
|
public static async UniTask<TSource> LastAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken, bool defaultIfEmpty)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -108,7 +108,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
public static async UniTask<TSource> LastAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -151,7 +151,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
public static async UniTask<TSource> LastAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
@@ -194,7 +194,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
public static async UniTask<TSource> LastAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
||||||
{
|
{
|
||||||
var e = source.GetAsyncEnumerator(cancellationToken);
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
||||||
try
|
try
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return LongCount.InvokeAsync(source, cancellationToken);
|
return LongCount.LongCountAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<long> LongCountAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<long> LongCountAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -18,7 +18,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return LongCount.InvokeAsync(source, predicate, cancellationToken);
|
return LongCount.LongCountAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<long> LongCountAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<long> LongCountAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -26,7 +26,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return LongCount.InvokeAsync(source, predicate, cancellationToken);
|
return LongCount.LongCountAwaitAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<long> LongCountAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
public static UniTask<long> LongCountAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
||||||
@@ -34,13 +34,13 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
||||||
|
|
||||||
return LongCount.InvokeAsync(source, predicate, cancellationToken);
|
return LongCount.LongCountAwaitWithCancellationAsync(source, predicate, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static class LongCount
|
internal static class LongCount
|
||||||
{
|
{
|
||||||
internal static async UniTask<long> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
|
internal static async UniTask<long> LongCountAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
|
|
||||||
@@ -63,7 +63,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<long> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<long> LongCountAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
|
|
||||||
@@ -89,7 +89,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<long> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<long> LongCountAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
|
|
||||||
@@ -115,7 +115,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static async UniTask<long> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
internal static async UniTask<long> LongCountAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
long count = 0;
|
long count = 0;
|
||||||
|
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Max.InvokeAsync(source, cancellationToken);
|
return Max.MaxAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TResult> MaxAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TResult> selector, CancellationToken cancellationToken = default)
|
public static UniTask<TResult> MaxAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TResult> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -19,7 +19,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Max.InvokeAsync(source, selector, cancellationToken);
|
return Max.MaxAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TResult> MaxAwaitAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TResult>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<TResult> MaxAwaitAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TResult>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -27,21 +27,21 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Max.InvokeAsync(source, selector, cancellationToken);
|
return Max.MaxAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TResult> MaxAwaitCancellationAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TResult>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<TResult> MaxAwaitWithCancellationAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TResult>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Max.InvokeAsync(source, selector, cancellationToken);
|
return Max.MaxAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static partial class Max
|
internal static partial class Max
|
||||||
{
|
{
|
||||||
public static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
|
public static async UniTask<TSource> MaxAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
TSource value = default;
|
TSource value = default;
|
||||||
var comparer = Comparer<TSource>.Default;
|
var comparer = Comparer<TSource>.Default;
|
||||||
@@ -80,7 +80,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TResult> InvokeAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TResult> selector, CancellationToken cancellationToken)
|
public static async UniTask<TResult> MaxAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TResult> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
TResult value = default;
|
TResult value = default;
|
||||||
var comparer = Comparer<TResult>.Default;
|
var comparer = Comparer<TResult>.Default;
|
||||||
@@ -119,7 +119,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TResult> InvokeAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TResult>> selector, CancellationToken cancellationToken)
|
public static async UniTask<TResult> MaxAwaitAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TResult>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
TResult value = default;
|
TResult value = default;
|
||||||
var comparer = Comparer<TResult>.Default;
|
var comparer = Comparer<TResult>.Default;
|
||||||
@@ -158,7 +158,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TResult> InvokeAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TResult>> selector, CancellationToken cancellationToken)
|
public static async UniTask<TResult> MaxAwaitWithCancellationAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TResult>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
TResult value = default;
|
TResult value = default;
|
||||||
var comparer = Comparer<TResult>.Default;
|
var comparer = Comparer<TResult>.Default;
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return Min.InvokeAsync(source, cancellationToken);
|
return Min.MinAsync(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TResult> MinAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TResult> selector, CancellationToken cancellationToken = default)
|
public static UniTask<TResult> MinAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TResult> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -19,7 +19,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Min.InvokeAsync(source, selector, cancellationToken);
|
return Min.MinAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TResult> MinAwaitAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TResult>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<TResult> MinAwaitAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TResult>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -27,21 +27,21 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Min.InvokeAsync(source, selector, cancellationToken);
|
return Min.MinAwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<TResult> MinAwaitCancellationAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TResult>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<TResult> MinAwaitWithCancellationAsync<TSource, TResult>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TResult>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return Min.InvokeAsync(source, selector, cancellationToken);
|
return Min.MinAwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal static partial class Min
|
internal static partial class Min
|
||||||
{
|
{
|
||||||
public static async UniTask<TSource> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
|
public static async UniTask<TSource> MinAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
TSource value = default;
|
TSource value = default;
|
||||||
var comparer = Comparer<TSource>.Default;
|
var comparer = Comparer<TSource>.Default;
|
||||||
@@ -80,7 +80,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TResult> InvokeAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TResult> selector, CancellationToken cancellationToken)
|
public static async UniTask<TResult> MinAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TResult> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
TResult value = default;
|
TResult value = default;
|
||||||
var comparer = Comparer<TResult>.Default;
|
var comparer = Comparer<TResult>.Default;
|
||||||
@@ -119,7 +119,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TResult> InvokeAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TResult>> selector, CancellationToken cancellationToken)
|
public static async UniTask<TResult> MinAwaitAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TResult>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
TResult value = default;
|
TResult value = default;
|
||||||
var comparer = Comparer<TResult>.Default;
|
var comparer = Comparer<TResult>.Default;
|
||||||
@@ -158,7 +158,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<TResult> InvokeAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TResult>> selector, CancellationToken cancellationToken)
|
public static async UniTask<TResult> MinAwaitWithCancellationAsync<TSource, TResult>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TResult>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
TResult value = default;
|
TResult value = default;
|
||||||
var comparer = Comparer<TResult>.Default;
|
var comparer = Comparer<TResult>.Default;
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -38,7 +38,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
return <#= minMax #>.InvokeAsync(source, cancellationToken);
|
return <#= minMax #>.<#= minMax #>Async(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<<#= TypeName(t) #>> <#= minMax #>Async<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, <#= TypeName(t) #>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<<#= TypeName(t) #>> <#= minMax #>Async<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, <#= TypeName(t) #>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -46,7 +46,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return <#= minMax #>.InvokeAsync(source, selector, cancellationToken);
|
return <#= minMax #>.<#= minMax #>Async(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<<#= TypeName(t) #>> <#= minMax #>AwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<<#= TypeName(t) #>> <#= minMax #>AwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken = default)
|
||||||
@@ -54,15 +54,15 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return <#= minMax #>.InvokeAsync(source, selector, cancellationToken);
|
return <#= minMax #>.<#= minMax #>AwaitAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static UniTask<<#= TypeName(t) #>> <#= minMax #>AwaitCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken = default)
|
public static UniTask<<#= TypeName(t) #>> <#= minMax #>AwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
Error.ThrowArgumentNullException(source, nameof(source));
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
Error.ThrowArgumentNullException(source, nameof(selector));
|
Error.ThrowArgumentNullException(source, nameof(selector));
|
||||||
|
|
||||||
return <#= minMax #>.InvokeAsync(source, selector, cancellationToken);
|
return <#= minMax #>.<#= minMax #>AwaitWithCancellationAsync(source, selector, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
<# } #>
|
<# } #>
|
||||||
@@ -71,7 +71,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
internal static partial class <#= minMax #>
|
internal static partial class <#= minMax #>
|
||||||
{
|
{
|
||||||
<# foreach(var t in types) { #>
|
<# foreach(var t in types) { #>
|
||||||
public static async UniTask<<#= TypeName(t) #>> InvokeAsync(IUniTaskAsyncEnumerable<<#= TypeName(t) #>> source, CancellationToken cancellationToken)
|
public static async UniTask<<#= TypeName(t) #>> <#= minMax #>Async(IUniTaskAsyncEnumerable<<#= TypeName(t) #>> source, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
<#= TypeName(t) #> value = default;
|
<#= TypeName(t) #> value = default;
|
||||||
|
|
||||||
@@ -117,7 +117,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<<#= TypeName(t) #>> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, <#= TypeName(t) #>> selector, CancellationToken cancellationToken)
|
public static async UniTask<<#= TypeName(t) #>> <#= minMax #>Async<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, <#= TypeName(t) #>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
<#= TypeName(t) #> value = default;
|
<#= TypeName(t) #> value = default;
|
||||||
|
|
||||||
@@ -163,7 +163,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<<#= TypeName(t) #>> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken)
|
public static async UniTask<<#= TypeName(t) #>> <#= minMax #>AwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
<#= TypeName(t) #> value = default;
|
<#= TypeName(t) #> value = default;
|
||||||
|
|
||||||
@@ -209,7 +209,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async UniTask<<#= TypeName(t) #>> InvokeAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken)
|
public static async UniTask<<#= TypeName(t) #>> <#= minMax #>AwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<<#= TypeName(t) #>>> selector, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
<#= TypeName(t) #> value = default;
|
<#= TypeName(t) #> value = default;
|
||||||
|
|
||||||
|
|||||||
@@ -20,14 +20,14 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(cancellationToken);
|
return new _Never(cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : IUniTaskAsyncEnumerator<T>
|
class _Never : IUniTaskAsyncEnumerator<T>
|
||||||
{
|
{
|
||||||
CancellationToken cancellationToken;
|
CancellationToken cancellationToken;
|
||||||
|
|
||||||
public Enumerator(CancellationToken cancellationToken)
|
public _Never(CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -25,12 +25,12 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TResult> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(source, cancellationToken);
|
return new _OfType(source, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : AsyncEnumeratorBase<object, TResult>
|
class _OfType : AsyncEnumeratorBase<object, TResult>
|
||||||
{
|
{
|
||||||
public Enumerator(IUniTaskAsyncEnumerable<object> source, CancellationToken cancellationToken)
|
public _OfType(IUniTaskAsyncEnumerable<object> source, CancellationToken cancellationToken)
|
||||||
|
|
||||||
: base(source, cancellationToken)
|
: base(source, cancellationToken)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -407,10 +407,10 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public IUniTaskAsyncEnumerator<TElement> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
public IUniTaskAsyncEnumerator<TElement> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
return new Enumerator(this, cancellationToken);
|
return new _OrderedAsyncEnumerator(this, cancellationToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator<TElement>
|
class _OrderedAsyncEnumerator : MoveNextSource, IUniTaskAsyncEnumerator<TElement>
|
||||||
{
|
{
|
||||||
protected readonly OrderedAsyncEnumerable<TElement> parent;
|
protected readonly OrderedAsyncEnumerable<TElement> parent;
|
||||||
CancellationToken cancellationToken;
|
CancellationToken cancellationToken;
|
||||||
@@ -418,10 +418,11 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
int[] map;
|
int[] map;
|
||||||
int index;
|
int index;
|
||||||
|
|
||||||
public Enumerator(OrderedAsyncEnumerable<TElement> parent, CancellationToken cancellationToken)
|
public _OrderedAsyncEnumerator(OrderedAsyncEnumerable<TElement> parent, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
this.parent = parent;
|
this.parent = parent;
|
||||||
this.cancellationToken = cancellationToken;
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TElement Current { get; private set; }
|
public TElement Current { get; private set; }
|
||||||
@@ -477,6 +478,7 @@ namespace Cysharp.Threading.Tasks.Linq
|
|||||||
|
|
||||||
public UniTask DisposeAsync()
|
public UniTask DisposeAsync()
|
||||||
{
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
return default;
|
return default;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
128
src/UniTask/Assets/Plugins/UniTask/Runtime/Linq/Pairwise.cs
Normal file
128
src/UniTask/Assets/Plugins/UniTask/Runtime/Linq/Pairwise.cs
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
using Cysharp.Threading.Tasks.Internal;
|
||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace Cysharp.Threading.Tasks.Linq
|
||||||
|
{
|
||||||
|
public static partial class UniTaskAsyncEnumerable
|
||||||
|
{
|
||||||
|
public static IUniTaskAsyncEnumerable<(TSource, TSource)> Pairwise<TSource>(this IUniTaskAsyncEnumerable<TSource> source)
|
||||||
|
{
|
||||||
|
Error.ThrowArgumentNullException(source, nameof(source));
|
||||||
|
|
||||||
|
return new Pairwise<TSource>(source);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal sealed class Pairwise<TSource> : IUniTaskAsyncEnumerable<(TSource, TSource)>
|
||||||
|
{
|
||||||
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
|
|
||||||
|
public Pairwise(IUniTaskAsyncEnumerable<TSource> source)
|
||||||
|
{
|
||||||
|
this.source = source;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IUniTaskAsyncEnumerator<(TSource, TSource)> GetAsyncEnumerator(CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
return new _Pairwise(source, cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
sealed class _Pairwise : MoveNextSource, IUniTaskAsyncEnumerator<(TSource, TSource)>
|
||||||
|
{
|
||||||
|
static readonly Action<object> MoveNextCoreDelegate = MoveNextCore;
|
||||||
|
|
||||||
|
readonly IUniTaskAsyncEnumerable<TSource> source;
|
||||||
|
CancellationToken cancellationToken;
|
||||||
|
|
||||||
|
IUniTaskAsyncEnumerator<TSource> enumerator;
|
||||||
|
UniTask<bool>.Awaiter awaiter;
|
||||||
|
|
||||||
|
TSource prev;
|
||||||
|
bool isFirst;
|
||||||
|
|
||||||
|
public _Pairwise(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
this.source = source;
|
||||||
|
this.cancellationToken = cancellationToken;
|
||||||
|
TaskTracker.TrackActiveTask(this, 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
public (TSource, TSource) Current { get; private set; }
|
||||||
|
|
||||||
|
public UniTask<bool> MoveNextAsync()
|
||||||
|
{
|
||||||
|
cancellationToken.ThrowIfCancellationRequested();
|
||||||
|
|
||||||
|
if (enumerator == null)
|
||||||
|
{
|
||||||
|
isFirst = true;
|
||||||
|
enumerator = source.GetAsyncEnumerator(cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
completionSource.Reset();
|
||||||
|
SourceMoveNext();
|
||||||
|
return new UniTask<bool>(this, completionSource.Version);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SourceMoveNext()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
awaiter = enumerator.MoveNextAsync().GetAwaiter();
|
||||||
|
if (awaiter.IsCompleted)
|
||||||
|
{
|
||||||
|
MoveNextCore(this);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
awaiter.SourceOnCompleted(MoveNextCoreDelegate, this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
completionSource.TrySetException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void MoveNextCore(object state)
|
||||||
|
{
|
||||||
|
var self = (_Pairwise)state;
|
||||||
|
|
||||||
|
if (self.TryGetResult(self.awaiter, out var result))
|
||||||
|
{
|
||||||
|
if (result)
|
||||||
|
{
|
||||||
|
if (self.isFirst)
|
||||||
|
{
|
||||||
|
self.isFirst = false;
|
||||||
|
self.prev = self.enumerator.Current;
|
||||||
|
self.SourceMoveNext(); // run again. okay to use recursive(only one more).
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
var p = self.prev;
|
||||||
|
self.prev = self.enumerator.Current;
|
||||||
|
self.Current = (p, self.prev);
|
||||||
|
self.completionSource.TrySetResult(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
self.completionSource.TrySetResult(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public UniTask DisposeAsync()
|
||||||
|
{
|
||||||
|
TaskTracker.RemoveTracking(this);
|
||||||
|
if (enumerator != null)
|
||||||
|
{
|
||||||
|
return enumerator.DisposeAsync();
|
||||||
|
}
|
||||||
|
return default;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
fileFormatVersion: 2
|
||||||
|
guid: cddbf051d2a88f549986c468b23214af
|
||||||
|
MonoImporter:
|
||||||
|
externalObjects: {}
|
||||||
|
serializedVersion: 2
|
||||||
|
defaultReferences: []
|
||||||
|
executionOrder: 0
|
||||||
|
icon: {instanceID: 0}
|
||||||
|
userData:
|
||||||
|
assetBundleName:
|
||||||
|
assetBundleVariant:
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user