Compare commits
	
		
			41 Commits
		
	
	
		
			v3.1.0
			...
			archive-pd
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | bcbb2f3699 | ||
|  | 963aa9f3e5 | ||
|  | e4c0c569ec | ||
|  | b6d0a9cdbd | ||
|  | eff23a770a | ||
|  | 34ce3ac803 | ||
|  | c59fd6df20 | ||
|  | 4824b20f28 | ||
|  | 4e53fa0392 | ||
|  | d478ed5608 | ||
|  | 8db649ba5f | ||
|  | e6390c8c41 | ||
|  | 2fec0b6dec | ||
|  | dd47c7c66d | ||
|  | 98855b4d84 | ||
|  | ac2d064cb0 | ||
|  | d63ed30966 | ||
|  | 7716eadca3 | ||
|  | e32d001809 | ||
|  | 2cb9813eb6 | ||
|  | af380116a0 | ||
|  | 3d3ae7129d | ||
|  | b44defe60a | ||
|  | dfad394be0 | ||
|  | a3ad6afee2 | ||
|  | 7c41a8ec19 | ||
|  | 5df1c5ae6d | ||
|  | c05c7148dd | ||
|  | ba9fe00c4d | ||
|  | cac0362377 | ||
|  | 7e0480e2ef | ||
|  | 566076e3d8 | ||
|  | aa02012c14 | ||
|  | f80bd72d22 | ||
|  | acf7d6a054 | ||
|  | 1129c2ce4f | ||
|  | 64027b336f | ||
|  | 2ce9fe068b | ||
|  | be8a8e9380 | ||
|  | c70acbe7d1 | ||
|  | d97e6b48ed | 
| @@ -43,5 +43,5 @@ before_install: | |||||||
| script: | script: | ||||||
|     - mkdir build |     - mkdir build | ||||||
|     - cd build |     - cd build | ||||||
|     - cmake -DCLANG_FORMAT_SUFFIX=$CLANG_FORMAT_SUFFIX --config Release .. |     - cmake -DCLANG_FORMAT_SUFFIX=$CLANG_FORMAT_SUFFIX -DWARNINGS_AS_ERRORS=On --config Release .. | ||||||
|     - cmake --build . -- -j2 |     - cmake --build . -- -j2 | ||||||
|   | |||||||
							
								
								
									
										95
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										95
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,5 +1,11 @@ | |||||||
| # Discord RPC | # Discord RPC | ||||||
|  |  | ||||||
|  | ## Deprecation Notice | ||||||
|  |  | ||||||
|  | This library has been deprecated in favor of Discord's GameSDK. [Learn more here](https://discordapp.com/developers/docs/game-sdk/sdk-starter-guide) | ||||||
|  |  | ||||||
|  | --- | ||||||
|  |  | ||||||
| This is a library for interfacing your game with a locally running Discord desktop client. It's known to work on Windows, macOS, and Linux. You can use the lib directly if you like, or use it as a guide to writing your own if it doesn't suit your game as is. PRs/feedback welcome if you have an improvement everyone might want, or can describe how this doesn't meet your needs. | This is a library for interfacing your game with a locally running Discord desktop client. It's known to work on Windows, macOS, and Linux. You can use the lib directly if you like, or use it as a guide to writing your own if it doesn't suit your game as is. PRs/feedback welcome if you have an improvement everyone might want, or can describe how this doesn't meet your needs. | ||||||
|  |  | ||||||
| Included here are some quick demos that implement the very minimal subset to show current status, and | Included here are some quick demos that implement the very minimal subset to show current status, and | ||||||
| @@ -15,13 +21,76 @@ Zeroith, you should be set up to build things because you are a game developer, | |||||||
|  |  | ||||||
| First, head on over to the [Discord developers site](https://discordapp.com/developers/applications/me) and make yourself an app. Keep track of `Client ID` -- you'll need it here to pass to the init function. | First, head on over to the [Discord developers site](https://discordapp.com/developers/applications/me) and make yourself an app. Keep track of `Client ID` -- you'll need it here to pass to the init function. | ||||||
|  |  | ||||||
|  | ### Unreal Engine 4 Setup | ||||||
|  |  | ||||||
|  | To use the Rich Presense plugin with Unreal Engine Projects: | ||||||
|  |  | ||||||
|  | 1.  Download the latest [release](https://github.com/discordapp/discord-rpc/releases) for each operating system you are targeting and the zipped source code | ||||||
|  | 2.  In the source code zip, copy the UE plugin—`examples/unrealstatus/Plugins/discordrpc`—to your project's plugin directory | ||||||
|  | 3.  At `[YOUR_UE_PROJECT]/Plugins/discordrpc/source/ThirdParty/DiscordRpcLibrary/`, create an `Include` folder and copy `discord_rpc.h` and `discord_register.h` to it from the zip | ||||||
|  | 4.  Follow the steps below for each OS | ||||||
|  | 5.  Build your UE4 project | ||||||
|  | 6.  Launch the editor, and enable the Discord plugin. | ||||||
|  |  | ||||||
|  | #### Windows | ||||||
|  |  | ||||||
|  | - At `[YOUR_UE_PROJECT]/Plugins/discordrpc/source/ThirdParty/DiscordRpcLibrary/`, create a `Win64` folder | ||||||
|  | - Copy `lib/discord-rpc.lib` and `bin/discord-rpc.dll` from `[RELEASE_ZIP]/win64-dynamic` to the `Win64` folder | ||||||
|  |  | ||||||
|  | #### Mac | ||||||
|  |  | ||||||
|  | - At `[YOUR_UE_PROJECT]/Plugins/discordrpc/source/ThirdParty/DiscordRpcLibrary/`, create a `Mac` folder | ||||||
|  | - Copy `libdiscord-rpc.dylib` from `[RELEASE_ZIP]/osx-dynamic/lib` to the `Mac` folder | ||||||
|  |  | ||||||
|  | #### Linux | ||||||
|  |  | ||||||
|  | - At `[YOUR_UE_PROJECT]/Plugins/discordrpc/source/ThirdParty/DiscordRpcLibrary/`, create a `Linux` folder | ||||||
|  | - Inside, create another folder `x86_64-unknown-linux-gnu` | ||||||
|  | - Copy `libdiscord-rpc.so` from `[RELEASE_ZIP]/linux-dynamic/lib` to `Linux/x86_64-unknown-linux-gnu` | ||||||
|  |  | ||||||
|  | ### Unity Setup | ||||||
|  |  | ||||||
|  | If you're a Unity developer looking to integrate Rich Presence into your game, follow this simple guide to get started towards success: | ||||||
|  |  | ||||||
|  | 1. Download the DLLs for any platform that you need from [our releases](https://github.com/discordapp/discord-rpc/releases) | ||||||
|  | 2. In your Unity project, create a `Plugins` folder inside your `Assets` folder if you don't already have one | ||||||
|  | 3. Copy the file `DiscordRpc.cs` from [here](https://github.com/discordapp/discord-rpc/blob/master/examples/button-clicker/Assets/DiscordRpc.cs) into your `Assets` folder. This is basically your header file for the SDK | ||||||
|  |  | ||||||
|  | We've got our `Plugins` folder ready, so let's get platform-specific! | ||||||
|  |  | ||||||
|  | #### Windows | ||||||
|  |  | ||||||
|  | 4. Create `x86` and `x86_64` folders inside `Assets/Plugins/` | ||||||
|  | 5. Copy `discord-rpc-win/win64-dynamic/bin/discord-rpc.dll` to `Assets/Plugins/x86_64/` | ||||||
|  | 6. Copy `discord-rpc-win/win32-dynamic/bin/discord-rpc.dll` to `Assets/Plugins/x86/` | ||||||
|  | 7. Click on both DLLs and make sure they are targetting the correct architectures in the Unity editor properties pane | ||||||
|  | 8. Done! | ||||||
|  |  | ||||||
|  | #### MacOS | ||||||
|  |  | ||||||
|  | 4. Copy `discord-rpc-osx/osx-dynamic/lib/libdiscord-rpc.dylib` to `Assets/Plugins/` | ||||||
|  | 5. Rename `libdiscord-rpc.dylib` to `discord-rpc.bundle` | ||||||
|  | 6. Done! | ||||||
|  |  | ||||||
|  | #### Linux | ||||||
|  |  | ||||||
|  | 4. Copy `discord-rpc-linux/linux-dynamic-lib/libdiscord-rpc.so` to `Assets/Plugins/` | ||||||
|  | 5. Done! | ||||||
|  |  | ||||||
|  | You're ready to roll! For code examples on how to interact with the SDK using the `DiscordRpc.cs` header file, check out [our example](https://github.com/discordapp/discord-rpc/blob/master/examples/button-clicker/Assets/DiscordController.cs) | ||||||
|  |  | ||||||
| ### From package | ### From package | ||||||
|  |  | ||||||
| Download a release package for your platform(s) -- they have subdirs with various prebuilt options, select the one you need add `/include` to your compile includes, `/lib` to your linker paths, and link with `discord-rpc`. For the dynamically linked builds, you'll need to ship the associated file along with your game. | Download a release package for your platform(s) -- they have subdirs with various prebuilt options, select the one you need add `/include` to your compile includes, `/lib` to your linker paths, and link with `discord-rpc`. For the dynamically linked builds, you'll need to ship the associated file along with your game. | ||||||
|  |  | ||||||
| ### From repo | ### From repo | ||||||
|  |  | ||||||
|  | First-eth, you'll want `CMake`. There's a few different ways to install it on your system, and you should refer to [their website](https://cmake.org/install/). Many package managers provide ways of installing CMake as well. | ||||||
|  |  | ||||||
|  | To make sure it's installed correctly, type `cmake --version` into your flavor of terminal/cmd. If you get a response with a version number, you're good to go! | ||||||
|  |  | ||||||
| There's a [CMake](https://cmake.org/download/) file that should be able to generate the lib for you; Sometimes I use it like this: | There's a [CMake](https://cmake.org/download/) file that should be able to generate the lib for you; Sometimes I use it like this: | ||||||
|  |  | ||||||
| ```sh | ```sh | ||||||
|     cd <path to discord-rpc> |     cd <path to discord-rpc> | ||||||
|     mkdir build |     mkdir build | ||||||
| @@ -29,6 +98,7 @@ There's a [CMake](https://cmake.org/download/) file that should be able to gener | |||||||
|     cmake .. -DCMAKE_INSTALL_PREFIX=<path to install discord-rpc to> |     cmake .. -DCMAKE_INSTALL_PREFIX=<path to install discord-rpc to> | ||||||
|     cmake --build . --config Release --target install |     cmake --build . --config Release --target install | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| There is a wrapper build script `build.py` that runs `cmake` with a few different options. | There is a wrapper build script `build.py` that runs `cmake` with a few different options. | ||||||
|  |  | ||||||
| Usually, I run `build.py` to get things started, then use the generated project files as I work on things. It does depend on `click` library, so do a quick `pip install click` to make sure you have it if you want to run `build.py`. | Usually, I run `build.py` to get things started, then use the generated project files as I work on things. It does depend on `click` library, so do a quick `pip install click` to make sure you have it if you want to run `build.py`. | ||||||
| @@ -36,20 +106,21 @@ Usually, I run `build.py` to get things started, then use the generated project | |||||||
| There are some CMake options you might care about: | There are some CMake options you might care about: | ||||||
|  |  | ||||||
| | flag                                                                                     | default | does                                                                                                                                                  | | | flag                                                                                     | default | does                                                                                                                                                  | | ||||||
| |------|---------|------| | | ---------------------------------------------------------------------------------------- | ------- | ----------------------------------------------------------------------------------------------------------------------------------------------------- | | ||||||
| | `ENABLE_IO_THREAD` | `ON` | When enabled, we start up a thread to do io processing, if disabled you should call `Discord_UpdateConnection` yourself. | | `ENABLE_IO_THREAD`                                                                       | `ON`    | When enabled, we start up a thread to do io processing, if disabled you should call `Discord_UpdateConnection` yourself.                              | | ||||||
| | `USE_STATIC_CRT` | `OFF` | (Windows) Enable to statically link the CRT, avoiding requiring users install the redistributable package. (The prebuilt binaries enable this option) | | `USE_STATIC_CRT`                                                                         | `OFF`   | (Windows) Enable to statically link the CRT, avoiding requiring users install the redistributable package. (The prebuilt binaries enable this option) | | ||||||
| | [`BUILD_SHARED_LIBS`](https://cmake.org/cmake/help/v3.7/variable/BUILD_SHARED_LIBS.html) | `OFF` | Build library as a DLL | | [`BUILD_SHARED_LIBS`](https://cmake.org/cmake/help/v3.7/variable/BUILD_SHARED_LIBS.html) | `OFF`   | Build library as a DLL                                                                                                                                | | ||||||
|  | | `WARNINGS_AS_ERRORS`                                                                     | `OFF`   | When enabled, compiles with `-Werror` (on \*nix platforms).                                                                                           | | ||||||
|  |  | ||||||
| ## Continuous Builds | ## Continuous Builds | ||||||
|  |  | ||||||
| Why do we have three of these? Three times the fun! | Why do we have three of these? Three times the fun! | ||||||
|  |  | ||||||
| | CI                   | badge                                                                                                                                            | | | CI                   | badge                                                                                                                                            | | ||||||
| |----|-------| | | -------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------ | | ||||||
| | TravisCI | [](https://travis-ci.org/discordapp/discord-rpc) | | TravisCI             | [](https://travis-ci.org/discordapp/discord-rpc)                  | | ||||||
| | AppVeyor | [](https://ci.appveyor.com/project/crmarsh/discord-rpc) | | AppVeyor             | [](https://ci.appveyor.com/project/crmarsh/discord-rpc)    | | ||||||
| | Buildkite (internal) | [](https://buildkite.com/discord/discord-rpc) | | Buildkite (internal) | [](https://buildkite.com/discord/discord-rpc) | | ||||||
|  |  | ||||||
| ## Sample: send-presence | ## Sample: send-presence | ||||||
|  |  | ||||||
| @@ -74,10 +145,14 @@ Below is a table of unofficial, community-developed wrappers for and implementat | |||||||
| ###### Rich Presence Wrappers and Implementations | ###### Rich Presence Wrappers and Implementations | ||||||
|  |  | ||||||
| | Name                                                                      | Language                          | | | Name                                                                      | Language                          | | ||||||
| |------|----------| | | ------------------------------------------------------------------------- | --------------------------------- | | ||||||
| | [discord-rpc.jar](https://github.com/Vatuu/discord-rpc "Discord-RPC.jar") | Java | | | [Discord RPC C#](https://github.com/Lachee/discord-rpc-csharp)            | C#                                | | ||||||
|  | | [Discord RPC D](https://github.com/voidblaster/discord-rpc-d)             | [D](https://dlang.org/)           | | ||||||
|  | | [discord-rpc.jar](https://github.com/Vatuu/discord-rpc 'Discord-RPC.jar') | Java                              | | ||||||
| | [java-discord-rpc](https://github.com/MinnDevelopment/java-discord-rpc)   | Java                              | | | [java-discord-rpc](https://github.com/MinnDevelopment/java-discord-rpc)   | Java                              | | ||||||
| | [Discord-IPC](https://github.com/jagrosh/DiscordIPC)                      | Java                              | | | [Discord-IPC](https://github.com/jagrosh/DiscordIPC)                      | Java                              | | ||||||
| | [Discord Rich Presence](https://npmjs.org/discord-rich-presence)          | JavaScript                        | | | [Discord Rich Presence](https://npmjs.org/discord-rich-presence)          | JavaScript                        | | ||||||
| | [drpc4k](https://github.com/Bluexin/drpc4k)                               | [Kotlin](https://kotlinlang.org/) | | | [drpc4k](https://github.com/Bluexin/drpc4k)                               | [Kotlin](https://kotlinlang.org/) | | ||||||
|  | | [lua-discordRPC](https://github.com/pfirsich/lua-discordRPC)              | LuaJIT (FFI)                      | | ||||||
|  | | [pypresence](https://github.com/qwertyquerty/pypresence)                  | [Python](https://python.org/)     | | ||||||
| | [SwordRPC](https://github.com/Azoy/SwordRPC)                              | [Swift](https://swift.org)        | | | [SwordRPC](https://github.com/Azoy/SwordRPC)                              | [Swift](https://swift.org)        | | ||||||
|   | |||||||
							
								
								
									
										59
									
								
								build.py
									
									
									
									
									
								
							
							
						
						
									
										59
									
								
								build.py
									
									
									
									
									
								
							| @@ -30,7 +30,7 @@ INSTALL_ROOT = os.path.join(SCRIPT_PATH, 'builds', 'install') | |||||||
| def get_signtool(): | def get_signtool(): | ||||||
|     """ get path to code signing tool """ |     """ get path to code signing tool """ | ||||||
|     if PLATFORM == 'win': |     if PLATFORM == 'win': | ||||||
|         sdk_dir = os.environ['WindowsSdkDir'] |         sdk_dir = 'c:\\Program Files (x86)\\Windows Kits\\10'  # os.environ['WindowsSdkDir'] | ||||||
|         return os.path.join(sdk_dir, 'bin', 'x86', 'signtool.exe') |         return os.path.join(sdk_dir, 'bin', 'x86', 'signtool.exe') | ||||||
|     elif PLATFORM == 'osx': |     elif PLATFORM == 'osx': | ||||||
|         return '/usr/bin/codesign' |         return '/usr/bin/codesign' | ||||||
| @@ -70,14 +70,7 @@ def cli(ctx, clean): | |||||||
| @click.pass_context | @click.pass_context | ||||||
| def unity(ctx): | def unity(ctx): | ||||||
|     """ build just dynamic libs for use in unity project """ |     """ build just dynamic libs for use in unity project """ | ||||||
|     ctx.invoke( |     ctx.invoke(libs, clean=False, static=False, shared=True, skip_formatter=True, just_release=True) | ||||||
|         libs, |  | ||||||
|         clean=False, |  | ||||||
|         static=False, |  | ||||||
|         shared=True, |  | ||||||
|         skip_formatter=True, |  | ||||||
|         just_release=True |  | ||||||
|     ) |  | ||||||
|     BUILDS = [] |     BUILDS = [] | ||||||
|  |  | ||||||
|     click.echo('--- Copying libs and header into unity example') |     click.echo('--- Copying libs and header into unity example') | ||||||
| @@ -85,11 +78,11 @@ def unity(ctx): | |||||||
|  |  | ||||||
|     if sys.platform.startswith('win'): |     if sys.platform.startswith('win'): | ||||||
|         LIBRARY_NAME = 'discord-rpc.dll' |         LIBRARY_NAME = 'discord-rpc.dll' | ||||||
|         BUILD_64_BASE_PATH = os.path.join(SCRIPT_PATH, 'builds', 'win64-dynamic', 'src', 'Release') |         BUILD_64_BASE_PATH = os.path.join(SCRIPT_PATH, 'builds', 'win64-dynamic', 'src', 'RelWithDebInfo') | ||||||
|         UNITY_64_DLL_PATH = os.path.join(UNITY_PROJECT_PATH, 'x86_64') |         UNITY_64_DLL_PATH = os.path.join(UNITY_PROJECT_PATH, 'x86_64') | ||||||
|         BUILDS.append({BUILD_64_BASE_PATH: UNITY_64_DLL_PATH}) |         BUILDS.append({BUILD_64_BASE_PATH: UNITY_64_DLL_PATH}) | ||||||
|  |  | ||||||
|         BUILD_32_BASE_PATH = os.path.join(SCRIPT_PATH, 'builds', 'win32-dynamic', 'src', 'Release') |         BUILD_32_BASE_PATH = os.path.join(SCRIPT_PATH, 'builds', 'win32-dynamic', 'src', 'RelWithDebInfo') | ||||||
|         UNITY_32_DLL_PATH = os.path.join(UNITY_PROJECT_PATH, 'x86') |         UNITY_32_DLL_PATH = os.path.join(UNITY_PROJECT_PATH, 'x86') | ||||||
|         BUILDS.append({BUILD_32_BASE_PATH: UNITY_32_DLL_PATH}) |         BUILDS.append({BUILD_32_BASE_PATH: UNITY_32_DLL_PATH}) | ||||||
|  |  | ||||||
| @@ -97,7 +90,8 @@ def unity(ctx): | |||||||
|         LIBRARY_NAME = 'discord-rpc.bundle' |         LIBRARY_NAME = 'discord-rpc.bundle' | ||||||
|         BUILD_BASE_PATH = os.path.join(SCRIPT_PATH, 'builds', 'osx-dynamic', 'src') |         BUILD_BASE_PATH = os.path.join(SCRIPT_PATH, 'builds', 'osx-dynamic', 'src') | ||||||
|         UNITY_DLL_PATH = UNITY_PROJECT_PATH |         UNITY_DLL_PATH = UNITY_PROJECT_PATH | ||||||
|         os.rename(os.path.join(BUILD_BASE_PATH, 'libdiscord-rpc.dylib'), os.path.join(BUILD_BASE_PATH, 'discord-rpc.bundle')) |         os.rename( | ||||||
|  |             os.path.join(BUILD_BASE_PATH, 'libdiscord-rpc.dylib'), os.path.join(BUILD_BASE_PATH, 'discord-rpc.bundle')) | ||||||
|  |  | ||||||
|         BUILDS.append({BUILD_BASE_PATH: UNITY_DLL_PATH}) |         BUILDS.append({BUILD_BASE_PATH: UNITY_DLL_PATH}) | ||||||
|  |  | ||||||
| @@ -122,14 +116,7 @@ def unity(ctx): | |||||||
| @click.pass_context | @click.pass_context | ||||||
| def unreal(ctx): | def unreal(ctx): | ||||||
|     """ build libs and copy them into the unreal project """ |     """ build libs and copy them into the unreal project """ | ||||||
|     ctx.invoke( |     ctx.invoke(libs, clean=False, static=False, shared=True, skip_formatter=True, just_release=True) | ||||||
|         libs, |  | ||||||
|         clean=False, |  | ||||||
|         static=False, |  | ||||||
|         shared=True, |  | ||||||
|         skip_formatter=True, |  | ||||||
|         just_release=True |  | ||||||
|     ) |  | ||||||
|     BUILDS = [] |     BUILDS = [] | ||||||
|  |  | ||||||
|     click.echo('--- Copying libs and header into unreal example') |     click.echo('--- Copying libs and header into unreal example') | ||||||
| @@ -140,11 +127,11 @@ def unreal(ctx): | |||||||
|  |  | ||||||
|     if sys.platform.startswith('win'): |     if sys.platform.startswith('win'): | ||||||
|         LIBRARY_NAME = 'discord-rpc.lib' |         LIBRARY_NAME = 'discord-rpc.lib' | ||||||
|         BUILD_64_BASE_PATH = os.path.join(SCRIPT_PATH, 'builds', 'win64-dynamic', 'src', 'Release') |         BUILD_64_BASE_PATH = os.path.join(SCRIPT_PATH, 'builds', 'win64-dynamic', 'src', 'RelWithDebInfo') | ||||||
|         UNREAL_64_DLL_PATH = os.path.join(UNREAL_PROJECT_PATH, 'Source', 'ThirdParty', 'DiscordRpcLibrary', 'Win64') |         UNREAL_64_DLL_PATH = os.path.join(UNREAL_PROJECT_PATH, 'Source', 'ThirdParty', 'DiscordRpcLibrary', 'Win64') | ||||||
|         BUILDS.append({BUILD_64_BASE_PATH: UNREAL_64_DLL_PATH}) |         BUILDS.append({BUILD_64_BASE_PATH: UNREAL_64_DLL_PATH}) | ||||||
|  |  | ||||||
|         BUILD_32_BASE_PATH = os.path.join(SCRIPT_PATH, 'builds', 'win32-dynamic', 'src', 'Release') |         BUILD_32_BASE_PATH = os.path.join(SCRIPT_PATH, 'builds', 'win32-dynamic', 'src', 'RelWithDebInfo') | ||||||
|         UNREAL_32_DLL_PATH = os.path.join(UNREAL_PROJECT_PATH, 'Source', 'ThirdParty', 'DiscordRpcLibrary', 'Win32') |         UNREAL_32_DLL_PATH = os.path.join(UNREAL_PROJECT_PATH, 'Source', 'ThirdParty', 'DiscordRpcLibrary', 'Win32') | ||||||
|         BUILDS.append({BUILD_32_BASE_PATH: UNREAL_32_DLL_PATH}) |         BUILDS.append({BUILD_32_BASE_PATH: UNREAL_32_DLL_PATH}) | ||||||
|  |  | ||||||
| @@ -178,11 +165,7 @@ def build_lib(build_name, generator, options, just_release): | |||||||
|     mkdir_p(build_path) |     mkdir_p(build_path) | ||||||
|     mkdir_p(install_path) |     mkdir_p(install_path) | ||||||
|     with cd(build_path): |     with cd(build_path): | ||||||
|         initial_cmake = [ |         initial_cmake = ['cmake', SCRIPT_PATH, '-DCMAKE_INSTALL_PREFIX=%s' % os.path.join('..', 'install', build_name)] | ||||||
|             'cmake', |  | ||||||
|             SCRIPT_PATH, |  | ||||||
|             '-DCMAKE_INSTALL_PREFIX=%s' % os.path.join('..', 'install', build_name) |  | ||||||
|         ] |  | ||||||
|         if generator: |         if generator: | ||||||
|             initial_cmake.extend(['-G', generator]) |             initial_cmake.extend(['-G', generator]) | ||||||
|         for key in options: |         for key in options: | ||||||
| @@ -194,7 +177,7 @@ def build_lib(build_name, generator, options, just_release): | |||||||
|         subprocess.check_call(initial_cmake) |         subprocess.check_call(initial_cmake) | ||||||
|         if not just_release: |         if not just_release: | ||||||
|             subprocess.check_call(['cmake', '--build', '.', '--config', 'Debug']) |             subprocess.check_call(['cmake', '--build', '.', '--config', 'Debug']) | ||||||
|         subprocess.check_call(['cmake', '--build', '.', '--config', 'Release', '--target', 'install']) |         subprocess.check_call(['cmake', '--build', '.', '--config', 'RelWithDebInfo', '--target', 'install']) | ||||||
|  |  | ||||||
|  |  | ||||||
| @cli.command() | @cli.command() | ||||||
| @@ -224,22 +207,28 @@ def sign(): | |||||||
|         sign_command_base = [ |         sign_command_base = [ | ||||||
|             tool, |             tool, | ||||||
|             'sign', |             'sign', | ||||||
|             '/n', 'Discord Inc.', |             '/n', | ||||||
|  |             'Discord Inc.', | ||||||
|             '/a', |             '/a', | ||||||
|             '/tr', 'http://timestamp.digicert.com/rfc3161', |             '/tr', | ||||||
|  |             'http://timestamp.digicert.com/rfc3161', | ||||||
|             '/as', |             '/as', | ||||||
|             '/td', 'sha256', |             '/td', | ||||||
|             '/fd', 'sha256', |             'sha256', | ||||||
|  |             '/fd', | ||||||
|  |             'sha256', | ||||||
|         ] |         ] | ||||||
|     elif PLATFORM == 'osx': |     elif PLATFORM == 'osx': | ||||||
|         signable_extensions.add('.dylib') |         signable_extensions.add('.dylib') | ||||||
|         sign_command_base = [ |         sign_command_base = [ | ||||||
|             tool, |             tool, | ||||||
|             '--keychain', os.path.expanduser('~/Library/Keychains/login.keychain'), |             '--keychain', | ||||||
|  |             os.path.expanduser('~/Library/Keychains/login.keychain'), | ||||||
|             '-vvvv', |             '-vvvv', | ||||||
|             '--deep', |             '--deep', | ||||||
|             '--force', |             '--force', | ||||||
|             '--sign', 'Developer ID Application: Hammer & Chisel Inc. (53Q6R32WPB)', |             '--sign', | ||||||
|  |             'Developer ID Application: Hammer & Chisel Inc. (53Q6R32WPB)', | ||||||
|         ] |         ] | ||||||
|     else: |     else: | ||||||
|         click.secho('Not signing things on this platform yet', fg='red') |         click.secho('Not signing things on this platform yet', fg='red') | ||||||
| @@ -286,6 +275,8 @@ def libs(clean, static, shared, skip_formatter, just_release): | |||||||
|  |  | ||||||
|     if IS_BUILD_MACHINE: |     if IS_BUILD_MACHINE: | ||||||
|         just_release = True |         just_release = True | ||||||
|  |         static_options['WARNINGS_AS_ERRORS'] = True | ||||||
|  |         dynamic_options['WARNINGS_AS_ERRORS'] = True | ||||||
|  |  | ||||||
|     if PLATFORM == 'win': |     if PLATFORM == 'win': | ||||||
|         generator32 = 'Visual Studio 14 2015' |         generator32 = 'Visual Studio 14 2015' | ||||||
|   | |||||||
| @@ -7,16 +7,15 @@ public class DiscordJoinEvent : UnityEngine.Events.UnityEvent<string> { } | |||||||
| public class DiscordSpectateEvent : UnityEngine.Events.UnityEvent<string> { } | public class DiscordSpectateEvent : UnityEngine.Events.UnityEvent<string> { } | ||||||
|  |  | ||||||
| [System.Serializable] | [System.Serializable] | ||||||
| public class DiscordJoinRequestEvent : UnityEngine.Events.UnityEvent<DiscordRpc.JoinRequest> { } | public class DiscordJoinRequestEvent : UnityEngine.Events.UnityEvent<DiscordRpc.DiscordUser> { } | ||||||
|  |  | ||||||
| public class DiscordController : MonoBehaviour | public class DiscordController : MonoBehaviour | ||||||
| { | { | ||||||
|     public DiscordRpc.RichPresence presence = new DiscordRpc.RichPresence(); |     public DiscordRpc.RichPresence presence = new DiscordRpc.RichPresence(); | ||||||
|     public string applicationId; |     public string applicationId; | ||||||
|     public string optionalSteamId; |     public string optionalSteamId; | ||||||
|     public int callbackCalls; |  | ||||||
|     public int clickCounter; |     public int clickCounter; | ||||||
|     public DiscordRpc.JoinRequest joinRequest; |     public DiscordRpc.DiscordUser joinRequest; | ||||||
|     public UnityEngine.Events.UnityEvent onConnect; |     public UnityEngine.Events.UnityEvent onConnect; | ||||||
|     public UnityEngine.Events.UnityEvent onDisconnect; |     public UnityEngine.Events.UnityEvent onDisconnect; | ||||||
|     public UnityEngine.Events.UnityEvent hasResponded; |     public UnityEngine.Events.UnityEvent hasResponded; | ||||||
| @@ -32,6 +31,11 @@ public class DiscordController : MonoBehaviour | |||||||
|         clickCounter++; |         clickCounter++; | ||||||
|  |  | ||||||
|         presence.details = string.Format("Button clicked {0} times", clickCounter); |         presence.details = string.Format("Button clicked {0} times", clickCounter); | ||||||
|  |         presence.joinSecret = "aSecret"; | ||||||
|  |         presence.partyId = "aPartyId"; | ||||||
|  |         presence.partySize = 1; | ||||||
|  |         presence.partyMax = 3; | ||||||
|  |         presence.partyPrivacy = DiscordRpc.PartyPrivacy.Public; | ||||||
|  |  | ||||||
|         DiscordRpc.UpdatePresence(presence); |         DiscordRpc.UpdatePresence(presence); | ||||||
|     } |     } | ||||||
| @@ -50,43 +54,37 @@ public class DiscordController : MonoBehaviour | |||||||
|         hasResponded.Invoke(); |         hasResponded.Invoke(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void ReadyCallback() |     public void ReadyCallback(ref DiscordRpc.DiscordUser connectedUser) | ||||||
|     { |     { | ||||||
|         ++callbackCalls; |         Debug.Log(string.Format("Discord: connected to {0}#{1}: {2}", connectedUser.username, connectedUser.discriminator, connectedUser.userId)); | ||||||
|         Debug.Log("Discord: ready"); |  | ||||||
|         onConnect.Invoke(); |         onConnect.Invoke(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void DisconnectedCallback(int errorCode, string message) |     public void DisconnectedCallback(int errorCode, string message) | ||||||
|     { |     { | ||||||
|         ++callbackCalls; |  | ||||||
|         Debug.Log(string.Format("Discord: disconnect {0}: {1}", errorCode, message)); |         Debug.Log(string.Format("Discord: disconnect {0}: {1}", errorCode, message)); | ||||||
|         onDisconnect.Invoke(); |         onDisconnect.Invoke(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void ErrorCallback(int errorCode, string message) |     public void ErrorCallback(int errorCode, string message) | ||||||
|     { |     { | ||||||
|         ++callbackCalls; |  | ||||||
|         Debug.Log(string.Format("Discord: error {0}: {1}", errorCode, message)); |         Debug.Log(string.Format("Discord: error {0}: {1}", errorCode, message)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void JoinCallback(string secret) |     public void JoinCallback(string secret) | ||||||
|     { |     { | ||||||
|         ++callbackCalls; |  | ||||||
|         Debug.Log(string.Format("Discord: join ({0})", secret)); |         Debug.Log(string.Format("Discord: join ({0})", secret)); | ||||||
|         onJoin.Invoke(secret); |         onJoin.Invoke(secret); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void SpectateCallback(string secret) |     public void SpectateCallback(string secret) | ||||||
|     { |     { | ||||||
|         ++callbackCalls; |  | ||||||
|         Debug.Log(string.Format("Discord: spectate ({0})", secret)); |         Debug.Log(string.Format("Discord: spectate ({0})", secret)); | ||||||
|         onSpectate.Invoke(secret); |         onSpectate.Invoke(secret); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void RequestCallback(ref DiscordRpc.JoinRequest request) |     public void RequestCallback(ref DiscordRpc.DiscordUser request) | ||||||
|     { |     { | ||||||
|         ++callbackCalls; |  | ||||||
|         Debug.Log(string.Format("Discord: join request {0}#{1}: {2}", request.username, request.discriminator, request.userId)); |         Debug.Log(string.Format("Discord: join request {0}#{1}: {2}", request.username, request.discriminator, request.userId)); | ||||||
|         joinRequest = request; |         joinRequest = request; | ||||||
|         onJoinRequest.Invoke(request); |         onJoinRequest.Invoke(request); | ||||||
| @@ -104,10 +102,8 @@ public class DiscordController : MonoBehaviour | |||||||
|     void OnEnable() |     void OnEnable() | ||||||
|     { |     { | ||||||
|         Debug.Log("Discord: init"); |         Debug.Log("Discord: init"); | ||||||
|         callbackCalls = 0; |  | ||||||
|  |  | ||||||
|         handlers = new DiscordRpc.EventHandlers(); |         handlers = new DiscordRpc.EventHandlers(); | ||||||
|         handlers.readyCallback = ReadyCallback; |         handlers.readyCallback += ReadyCallback; | ||||||
|         handlers.disconnectedCallback += DisconnectedCallback; |         handlers.disconnectedCallback += DisconnectedCallback; | ||||||
|         handlers.errorCallback += ErrorCallback; |         handlers.errorCallback += ErrorCallback; | ||||||
|         handlers.joinCallback += JoinCallback; |         handlers.joinCallback += JoinCallback; | ||||||
|   | |||||||
| @@ -2,35 +2,44 @@ | |||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Runtime.InteropServices; | using System.Runtime.InteropServices; | ||||||
| using System.Text; | using System.Text; | ||||||
|  | using AOT; | ||||||
|  |  | ||||||
| public class DiscordRpc | public class DiscordRpc | ||||||
| { | { | ||||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] |     [MonoPInvokeCallback(typeof(OnReadyInfo))] | ||||||
|     public delegate void ReadyCallback(); |     public static void ReadyCallback(ref DiscordUser connectedUser) { Callbacks.readyCallback(ref connectedUser); } | ||||||
|  |     public delegate void OnReadyInfo(ref DiscordUser connectedUser); | ||||||
|  |  | ||||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] |     [MonoPInvokeCallback(typeof(OnDisconnectedInfo))] | ||||||
|     public delegate void DisconnectedCallback(int errorCode, string message); |     public static void DisconnectedCallback(int errorCode, string message) { Callbacks.disconnectedCallback(errorCode, message); } | ||||||
|  |     public delegate void OnDisconnectedInfo(int errorCode, string message); | ||||||
|  |  | ||||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] |     [MonoPInvokeCallback(typeof(OnErrorInfo))] | ||||||
|     public delegate void ErrorCallback(int errorCode, string message); |     public static void ErrorCallback(int errorCode, string message) { Callbacks.errorCallback(errorCode, message); } | ||||||
|  |     public delegate void OnErrorInfo(int errorCode, string message); | ||||||
|  |  | ||||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] |     [MonoPInvokeCallback(typeof(OnJoinInfo))] | ||||||
|     public delegate void JoinCallback(string secret); |     public static void JoinCallback(string secret) { Callbacks.joinCallback(secret); } | ||||||
|  |     public delegate void OnJoinInfo(string secret); | ||||||
|  |  | ||||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] |     [MonoPInvokeCallback(typeof(OnSpectateInfo))] | ||||||
|     public delegate void SpectateCallback(string secret); |     public static void SpectateCallback(string secret) { Callbacks.spectateCallback(secret); } | ||||||
|  |     public delegate void OnSpectateInfo(string secret); | ||||||
|  |  | ||||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] |     [MonoPInvokeCallback(typeof(OnRequestInfo))] | ||||||
|     public delegate void RequestCallback(ref JoinRequest request); |     public static void RequestCallback(ref DiscordUser request) { Callbacks.requestCallback(ref request); } | ||||||
|  |     public delegate void OnRequestInfo(ref DiscordUser request); | ||||||
|  |  | ||||||
|  |     static EventHandlers Callbacks { get; set; } | ||||||
|  |  | ||||||
|     public struct EventHandlers |     public struct EventHandlers | ||||||
|     { |     { | ||||||
|         public ReadyCallback readyCallback; |         public OnReadyInfo readyCallback; | ||||||
|         public DisconnectedCallback disconnectedCallback; |         public OnDisconnectedInfo disconnectedCallback; | ||||||
|         public ErrorCallback errorCallback; |         public OnErrorInfo errorCallback; | ||||||
|         public JoinCallback joinCallback; |         public OnJoinInfo joinCallback; | ||||||
|         public SpectateCallback spectateCallback; |         public OnSpectateInfo spectateCallback; | ||||||
|         public RequestCallback requestCallback; |         public OnRequestInfo requestCallback; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     [Serializable, StructLayout(LayoutKind.Sequential)] |     [Serializable, StructLayout(LayoutKind.Sequential)] | ||||||
| @@ -47,6 +56,7 @@ public class DiscordRpc | |||||||
|         public IntPtr partyId; /* max 128 bytes */ |         public IntPtr partyId; /* max 128 bytes */ | ||||||
|         public int partySize; |         public int partySize; | ||||||
|         public int partyMax; |         public int partyMax; | ||||||
|  |         public int partyPrivacy; | ||||||
|         public IntPtr matchSecret; /* max 128 bytes */ |         public IntPtr matchSecret; /* max 128 bytes */ | ||||||
|         public IntPtr joinSecret; /* max 128 bytes */ |         public IntPtr joinSecret; /* max 128 bytes */ | ||||||
|         public IntPtr spectateSecret; /* max 128 bytes */ |         public IntPtr spectateSecret; /* max 128 bytes */ | ||||||
| @@ -54,7 +64,7 @@ public class DiscordRpc | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     [Serializable] |     [Serializable] | ||||||
|     public struct JoinRequest |     public struct DiscordUser | ||||||
|     { |     { | ||||||
|         public string userId; |         public string userId; | ||||||
|         public string username; |         public string username; | ||||||
| @@ -69,8 +79,29 @@ public class DiscordRpc | |||||||
|         Ignore = 2 |         Ignore = 2 | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public enum PartyPrivacy | ||||||
|  |     { | ||||||
|  |         Private = 0, | ||||||
|  |         Public = 1 | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static void Initialize(string applicationId, ref EventHandlers handlers, bool autoRegister, string optionalSteamId) | ||||||
|  |     { | ||||||
|  |         Callbacks = handlers; | ||||||
|  |  | ||||||
|  |         EventHandlers staticEventHandlers = new EventHandlers(); | ||||||
|  |         staticEventHandlers.readyCallback += DiscordRpc.ReadyCallback; | ||||||
|  |         staticEventHandlers.disconnectedCallback += DiscordRpc.DisconnectedCallback; | ||||||
|  |         staticEventHandlers.errorCallback += DiscordRpc.ErrorCallback; | ||||||
|  |         staticEventHandlers.joinCallback += DiscordRpc.JoinCallback; | ||||||
|  |         staticEventHandlers.spectateCallback += DiscordRpc.SpectateCallback; | ||||||
|  |         staticEventHandlers.requestCallback += DiscordRpc.RequestCallback; | ||||||
|  |  | ||||||
|  |         InitializeInternal(applicationId, ref staticEventHandlers, autoRegister, optionalSteamId); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     [DllImport("discord-rpc", EntryPoint = "Discord_Initialize", CallingConvention = CallingConvention.Cdecl)] |     [DllImport("discord-rpc", EntryPoint = "Discord_Initialize", CallingConvention = CallingConvention.Cdecl)] | ||||||
|     public static extern void Initialize(string applicationId, ref EventHandlers handlers, bool autoRegister, string optionalSteamId); |     static extern void InitializeInternal(string applicationId, ref EventHandlers handlers, bool autoRegister, string optionalSteamId); | ||||||
|  |  | ||||||
|     [DllImport("discord-rpc", EntryPoint = "Discord_Shutdown", CallingConvention = CallingConvention.Cdecl)] |     [DllImport("discord-rpc", EntryPoint = "Discord_Shutdown", CallingConvention = CallingConvention.Cdecl)] | ||||||
|     public static extern void Shutdown(); |     public static extern void Shutdown(); | ||||||
| @@ -113,6 +144,7 @@ public class DiscordRpc | |||||||
|         public string partyId; /* max 128 bytes */ |         public string partyId; /* max 128 bytes */ | ||||||
|         public int partySize; |         public int partySize; | ||||||
|         public int partyMax; |         public int partyMax; | ||||||
|  |         public PartyPrivacy partyPrivacy; | ||||||
|         public string matchSecret; /* max 128 bytes */ |         public string matchSecret; /* max 128 bytes */ | ||||||
|         public string joinSecret; /* max 128 bytes */ |         public string joinSecret; /* max 128 bytes */ | ||||||
|         public string spectateSecret; /* max 128 bytes */ |         public string spectateSecret; /* max 128 bytes */ | ||||||
| @@ -129,20 +161,21 @@ public class DiscordRpc | |||||||
|                 FreeMem(); |                 FreeMem(); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             _presence.state = StrToPtr(state, 128); |             _presence.state = StrToPtr(state); | ||||||
|             _presence.details = StrToPtr(details, 128); |             _presence.details = StrToPtr(details); | ||||||
|             _presence.startTimestamp = startTimestamp; |             _presence.startTimestamp = startTimestamp; | ||||||
|             _presence.endTimestamp = endTimestamp; |             _presence.endTimestamp = endTimestamp; | ||||||
|             _presence.largeImageKey = StrToPtr(largeImageKey, 32); |             _presence.largeImageKey = StrToPtr(largeImageKey); | ||||||
|             _presence.largeImageText = StrToPtr(largeImageText, 128); |             _presence.largeImageText = StrToPtr(largeImageText); | ||||||
|             _presence.smallImageKey = StrToPtr(smallImageKey, 32); |             _presence.smallImageKey = StrToPtr(smallImageKey); | ||||||
|             _presence.smallImageText = StrToPtr(smallImageText, 128); |             _presence.smallImageText = StrToPtr(smallImageText); | ||||||
|             _presence.partyId = StrToPtr(partyId, 128); |             _presence.partyId = StrToPtr(partyId); | ||||||
|             _presence.partySize = partySize; |             _presence.partySize = partySize; | ||||||
|             _presence.partyMax = partyMax; |             _presence.partyMax = partyMax; | ||||||
|             _presence.matchSecret = StrToPtr(matchSecret, 128); |             _presence.partyPrivacy = (int)partyPrivacy; | ||||||
|             _presence.joinSecret = StrToPtr(joinSecret, 128); |             _presence.matchSecret = StrToPtr(matchSecret); | ||||||
|             _presence.spectateSecret = StrToPtr(spectateSecret, 128); |             _presence.joinSecret = StrToPtr(joinSecret); | ||||||
|  |             _presence.spectateSecret = StrToPtr(spectateSecret); | ||||||
|             _presence.instance = instance; |             _presence.instance = instance; | ||||||
|  |  | ||||||
|             return _presence; |             return _presence; | ||||||
| @@ -152,16 +185,18 @@ public class DiscordRpc | |||||||
|         /// Returns a pointer to a representation of the given string with a size of maxbytes |         /// Returns a pointer to a representation of the given string with a size of maxbytes | ||||||
|         /// </summary> |         /// </summary> | ||||||
|         /// <param name="input">String to convert</param> |         /// <param name="input">String to convert</param> | ||||||
|         /// <param name="maxbytes">Max number of bytes to use</param> |  | ||||||
|         /// <returns>Pointer to the UTF-8 representation of <see cref="input"/></returns> |         /// <returns>Pointer to the UTF-8 representation of <see cref="input"/></returns> | ||||||
|         private IntPtr StrToPtr(string input, int maxbytes) |         private IntPtr StrToPtr(string input) | ||||||
|         { |         { | ||||||
|             if (string.IsNullOrEmpty(input)) return IntPtr.Zero; |             if (string.IsNullOrEmpty(input)) return IntPtr.Zero; | ||||||
|             var convstr = StrClampBytes(input, maxbytes); |             var convbytecnt = Encoding.UTF8.GetByteCount(input); | ||||||
|             var convbytecnt = Encoding.UTF8.GetByteCount(convstr); |             var buffer = Marshal.AllocHGlobal(convbytecnt + 1); | ||||||
|             var buffer = Marshal.AllocHGlobal(convbytecnt); |             for (int i = 0; i < convbytecnt + 1; i++) | ||||||
|  |             { | ||||||
|  |                 Marshal.WriteByte(buffer, i, 0); | ||||||
|  |             } | ||||||
|             _buffers.Add(buffer); |             _buffers.Add(buffer); | ||||||
|             Marshal.Copy(Encoding.UTF8.GetBytes(convstr), 0, buffer, convbytecnt); |             Marshal.Copy(Encoding.UTF8.GetBytes(input), 0, buffer, convbytecnt); | ||||||
|             return buffer; |             return buffer; | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -181,30 +216,6 @@ public class DiscordRpc | |||||||
|             return Encoding.UTF8.GetString(Encoding.UTF8.GetBytes(str)); |             return Encoding.UTF8.GetString(Encoding.UTF8.GetBytes(str)); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         /// <summary> |  | ||||||
|         /// Clamp the string to the given byte length preserving null termination |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="toclamp">string to clamp</param> |  | ||||||
|         /// <param name="maxbytes">max bytes the resulting string should have (including null termination)</param> |  | ||||||
|         /// <returns>null terminated string with a byte length less or equal to <see cref="maxbytes"/></returns> |  | ||||||
|         private static string StrClampBytes(string toclamp, int maxbytes) |  | ||||||
|         { |  | ||||||
|             var str = StrToUtf8NullTerm(toclamp); |  | ||||||
|             var strbytes = Encoding.UTF8.GetBytes(str); |  | ||||||
|  |  | ||||||
|             if (strbytes.Length <= maxbytes) |  | ||||||
|             { |  | ||||||
|                 return str; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             var newstrbytes = new byte[] { }; |  | ||||||
|             Array.Copy(strbytes, 0, newstrbytes, 0, maxbytes - 1); |  | ||||||
|             newstrbytes[newstrbytes.Length - 1] = 0; |  | ||||||
|             newstrbytes[newstrbytes.Length - 2] = 0; |  | ||||||
|  |  | ||||||
|             return Encoding.UTF8.GetString(newstrbytes); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Free the allocated memory for conversion to <see cref="RichPresenceStruct"/> |         /// Free the allocated memory for conversion to <see cref="RichPresenceStruct"/> | ||||||
|         /// </summary> |         /// </summary> | ||||||
|   | |||||||
| @@ -46,8 +46,8 @@ public class ScriptBatch | |||||||
| 		string[] srcDlls = { "../../builds/install/osx-dynamic/lib/libdiscord-rpc.dylib" }; | 		string[] srcDlls = { "../../builds/install/osx-dynamic/lib/libdiscord-rpc.dylib" }; | ||||||
| 		#else | 		#else | ||||||
| 		string[] dstDirs = { "Assets/Plugins", "Assets/Plugins/x86", "Assets/Plugins/x86_64" }; | 		string[] dstDirs = { "Assets/Plugins", "Assets/Plugins/x86", "Assets/Plugins/x86_64" }; | ||||||
| 		string[] dstDlls = { "Assets/Plugins/x86/discord-rpc.so", "Assets/Plugins/x86_64/discord-rpc.so" }; | 		string[] dstDlls = { "Assets/Plugins/discord-rpc.so" }; | ||||||
| 		string[] srcDlls = { "../../builds/install/linux-dynamic/bin/discord-rpc.dll", "../../builds/install/win64-dynamic/bin/discord-rpc.dll" }; | 		string[] srcDlls = { "../../builds/install/linux-dynamic/lib/libdiscord-rpc.so" }; | ||||||
| 		#endif | 		#endif | ||||||
|  |  | ||||||
| 		Debug.Assert(dstDlls.Length == srcDlls.Length); | 		Debug.Assert(dstDlls.Length == srcDlls.Length); | ||||||
|   | |||||||
| @@ -47,19 +47,24 @@ static void updateDiscordPresence() | |||||||
|         discordPresence.partyId = "party1234"; |         discordPresence.partyId = "party1234"; | ||||||
|         discordPresence.partySize = 1; |         discordPresence.partySize = 1; | ||||||
|         discordPresence.partyMax = 6; |         discordPresence.partyMax = 6; | ||||||
|  |         discordPresence.partyPrivacy = DISCORD_PARTY_PUBLIC; | ||||||
|         discordPresence.matchSecret = "xyzzy"; |         discordPresence.matchSecret = "xyzzy"; | ||||||
|         discordPresence.joinSecret = "join"; |         discordPresence.joinSecret = "join"; | ||||||
|         discordPresence.spectateSecret = "look"; |         discordPresence.spectateSecret = "look"; | ||||||
|         discordPresence.instance = 0; |         discordPresence.instance = 0; | ||||||
|         Discord_UpdatePresence(&discordPresence); |         Discord_UpdatePresence(&discordPresence); | ||||||
|     } else { |     } | ||||||
|  |     else { | ||||||
|         Discord_ClearPresence(); |         Discord_ClearPresence(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| static void handleDiscordReady(void) | static void handleDiscordReady(const DiscordUser* connectedUser) | ||||||
| { | { | ||||||
|     printf("\nDiscord: ready\n"); |     printf("\nDiscord: connected to user %s#%s - %s\n", | ||||||
|  |            connectedUser->username, | ||||||
|  |            connectedUser->discriminator, | ||||||
|  |            connectedUser->userId); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void handleDiscordDisconnected(int errcode, const char* message) | static void handleDiscordDisconnected(int errcode, const char* message) | ||||||
| @@ -82,13 +87,13 @@ static void handleDiscordSpectate(const char* secret) | |||||||
|     printf("\nDiscord: spectate (%s)\n", secret); |     printf("\nDiscord: spectate (%s)\n", secret); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void handleDiscordJoinRequest(const DiscordJoinRequest* request) | static void handleDiscordJoinRequest(const DiscordUser* request) | ||||||
| { | { | ||||||
|     int response = -1; |     int response = -1; | ||||||
|     char yn[4]; |     char yn[4]; | ||||||
|     printf("\nDiscord: join request from %s - %s - %s\n", |     printf("\nDiscord: join request from %s#%s - %s\n", | ||||||
|            request->username, |            request->username, | ||||||
|            request->avatar, |            request->discriminator, | ||||||
|            request->userId); |            request->userId); | ||||||
|     do { |     do { | ||||||
|         printf("Accept? (y/n)"); |         printf("Accept? (y/n)"); | ||||||
| @@ -152,7 +157,8 @@ static void gameLoop() | |||||||
|                 if (SendPresence) { |                 if (SendPresence) { | ||||||
|                     printf("Clearing presence information.\n"); |                     printf("Clearing presence information.\n"); | ||||||
|                     SendPresence = 0; |                     SendPresence = 0; | ||||||
|                 } else { |                 } | ||||||
|  |                 else { | ||||||
|                     printf("Restoring presence information.\n"); |                     printf("Restoring presence information.\n"); | ||||||
|                     SendPresence = 1; |                     SendPresence = 1; | ||||||
|                 } |                 } | ||||||
|   | |||||||
										
											Binary file not shown.
										
									
								
							| @@ -12,19 +12,26 @@ void FDiscordRpcModule::StartupModule() | |||||||
| #if defined(DISCORD_DYNAMIC_LIB) | #if defined(DISCORD_DYNAMIC_LIB) | ||||||
|     // Get the base directory of this plugin |     // Get the base directory of this plugin | ||||||
|     FString BaseDir = IPluginManager::Get().FindPlugin("DiscordRpc")->GetBaseDir(); |     FString BaseDir = IPluginManager::Get().FindPlugin("DiscordRpc")->GetBaseDir(); | ||||||
|     const FString SDKDir = FPaths::Combine(*BaseDir, TEXT("Source"), TEXT("ThirdParty"), TEXT("DiscordRpcLibrary")); |     const FString SDKDir = | ||||||
|  |       FPaths::Combine(*BaseDir, TEXT("Source"), TEXT("ThirdParty"), TEXT("DiscordRpcLibrary")); | ||||||
| #if PLATFORM_WINDOWS | #if PLATFORM_WINDOWS | ||||||
|     const FString LibName = TEXT("discord-rpc"); |     const FString LibName = TEXT("discord-rpc"); | ||||||
|     const FString LibDir = FPaths::Combine(*SDKDir, TEXT("Win64")); |     const FString LibDir = FPaths::Combine(*SDKDir, TEXT("Win64")); | ||||||
|     if (!LoadDependency(LibDir, LibName, DiscordRpcLibraryHandle)) { |     if (!LoadDependency(LibDir, LibName, DiscordRpcLibraryHandle)) { | ||||||
|         FMessageDialog::Open(EAppMsgType::Ok, LOCTEXT(LOCTEXT_NAMESPACE, "Failed to load DiscordRpc plugin. Plug-in will not be functional.")); |         FMessageDialog::Open( | ||||||
|  |           EAppMsgType::Ok, | ||||||
|  |           LOCTEXT(LOCTEXT_NAMESPACE, | ||||||
|  |                   "Failed to load DiscordRpc plugin. Plug-in will not be functional.")); | ||||||
|         FreeDependency(DiscordRpcLibraryHandle); |         FreeDependency(DiscordRpcLibraryHandle); | ||||||
|     } |     } | ||||||
| #elif PLATFORM_MAC | #elif PLATFORM_MAC | ||||||
|     const FString LibName = TEXT("libdiscord-rpc"); |     const FString LibName = TEXT("libdiscord-rpc"); | ||||||
|     const FString LibDir = FPaths::Combine(*SDKDir, TEXT("Mac")); |     const FString LibDir = FPaths::Combine(*SDKDir, TEXT("Mac")); | ||||||
|     if (!LoadDependency(LibDir, LibName, DiscordRpcLibraryHandle)) { |     if (!LoadDependency(LibDir, LibName, DiscordRpcLibraryHandle)) { | ||||||
|         FMessageDialog::Open(EAppMsgType::Ok, LOCTEXT(LOCTEXT_NAMESPACE, "Failed to load DiscordRpc plugin. Plug-in will not be functional.")); |         FMessageDialog::Open( | ||||||
|  |           EAppMsgType::Ok, | ||||||
|  |           LOCTEXT(LOCTEXT_NAMESPACE, | ||||||
|  |                   "Failed to load DiscordRpc plugin. Plug-in will not be functional.")); | ||||||
|         FreeDependency(DiscordRpcLibraryHandle); |         FreeDependency(DiscordRpcLibraryHandle); | ||||||
|     } |     } | ||||||
| #endif | #endif | ||||||
| @@ -49,8 +56,7 @@ bool FDiscordRpcModule::LoadDependency(const FString& Dir, const FString& Name, | |||||||
|  |  | ||||||
|     Handle = FPlatformProcess::GetDllHandle(*Path); |     Handle = FPlatformProcess::GetDllHandle(*Path); | ||||||
|  |  | ||||||
|     if (Handle == nullptr) |     if (Handle == nullptr) { | ||||||
|     { |  | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -59,8 +65,7 @@ bool FDiscordRpcModule::LoadDependency(const FString& Dir, const FString& Name, | |||||||
|  |  | ||||||
| void FDiscordRpcModule::FreeDependency(void*& Handle) | void FDiscordRpcModule::FreeDependency(void*& Handle) | ||||||
| { | { | ||||||
|     if (Handle != nullptr) |     if (Handle != nullptr) { | ||||||
|     { |  | ||||||
|         FPlatformProcess::FreeDllHandle(Handle); |         FPlatformProcess::FreeDllHandle(Handle); | ||||||
|         Handle = nullptr; |         Handle = nullptr; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -6,12 +6,22 @@ DEFINE_LOG_CATEGORY(Discord) | |||||||
|  |  | ||||||
| static UDiscordRpc* self = nullptr; | static UDiscordRpc* self = nullptr; | ||||||
|  |  | ||||||
| static void ReadyHandler() | static void ReadyHandler(const DiscordUser* connectedUser) | ||||||
| { | { | ||||||
|     UE_LOG(Discord, Log, TEXT("Discord connected")); |     FDiscordUserData ud; | ||||||
|  |     ud.userId = ANSI_TO_TCHAR(connectedUser->userId); | ||||||
|  |     ud.username = ANSI_TO_TCHAR(connectedUser->username); | ||||||
|  |     ud.discriminator = ANSI_TO_TCHAR(connectedUser->discriminator); | ||||||
|  |     ud.avatar = ANSI_TO_TCHAR(connectedUser->avatar); | ||||||
|  |     UE_LOG(Discord, | ||||||
|  |            Log, | ||||||
|  |            TEXT("Discord connected to %s - %s#%s"), | ||||||
|  |            *ud.userId, | ||||||
|  |            *ud.username, | ||||||
|  |            *ud.discriminator); | ||||||
|     if (self) { |     if (self) { | ||||||
|         self->IsConnected = true; |         self->IsConnected = true; | ||||||
|         self->OnConnected.Broadcast(); |         self->OnConnected.Broadcast(ud); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -52,16 +62,21 @@ static void SpectateGameHandler(const char* spectateSecret) | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| static void JoinRequestHandler(const DiscordJoinRequest* request) | static void JoinRequestHandler(const DiscordUser* request) | ||||||
| { | { | ||||||
|     FDiscordJoinRequestData jr; |     FDiscordUserData ud; | ||||||
|     jr.userId = ANSI_TO_TCHAR(request->userId); |     ud.userId = ANSI_TO_TCHAR(request->userId); | ||||||
|     jr.username = ANSI_TO_TCHAR(request->username); |     ud.username = ANSI_TO_TCHAR(request->username); | ||||||
|     jr.discriminator = ANSI_TO_TCHAR(request->discriminator); |     ud.discriminator = ANSI_TO_TCHAR(request->discriminator); | ||||||
|     jr.avatar = ANSI_TO_TCHAR(request->avatar); |     ud.avatar = ANSI_TO_TCHAR(request->avatar); | ||||||
|     UE_LOG(Discord, Log, TEXT("Discord join request from %s - %s#%s"), *jr.userId, *jr.username, *jr.discriminator); |     UE_LOG(Discord, | ||||||
|  |            Log, | ||||||
|  |            TEXT("Discord join request from %s - %s#%s"), | ||||||
|  |            *ud.userId, | ||||||
|  |            *ud.username, | ||||||
|  |            *ud.discriminator); | ||||||
|     if (self) { |     if (self) { | ||||||
|         self->OnJoinRequest.Broadcast(jr); |         self->OnJoinRequest.Broadcast(ud); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -138,6 +153,7 @@ void UDiscordRpc::UpdatePresence() | |||||||
|     rp.endTimestamp = RichPresence.endTimestamp; |     rp.endTimestamp = RichPresence.endTimestamp; | ||||||
|     rp.partySize = RichPresence.partySize; |     rp.partySize = RichPresence.partySize; | ||||||
|     rp.partyMax = RichPresence.partyMax; |     rp.partyMax = RichPresence.partyMax; | ||||||
|  |     rp.partyPrivacy = (int)RichPresence.partyPrivacy; | ||||||
|     rp.instance = RichPresence.instance; |     rp.instance = RichPresence.instance; | ||||||
|  |  | ||||||
|     Discord_UpdatePresence(&rp); |     Discord_UpdatePresence(&rp); | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ | |||||||
| * Ask to join callback data | * Ask to join callback data | ||||||
| */ | */ | ||||||
| USTRUCT(BlueprintType) | USTRUCT(BlueprintType) | ||||||
| struct FDiscordJoinRequestData { | struct FDiscordUserData { | ||||||
|     GENERATED_USTRUCT_BODY() |     GENERATED_USTRUCT_BODY() | ||||||
|  |  | ||||||
|     UPROPERTY(BlueprintReadOnly) |     UPROPERTY(BlueprintReadOnly) | ||||||
| @@ -24,15 +24,35 @@ struct FDiscordJoinRequestData { | |||||||
|     FString avatar; |     FString avatar; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  | * Valid response codes for Respond function | ||||||
|  | */ | ||||||
|  | UENUM(BlueprintType) | ||||||
|  | enum class EDiscordJoinResponseCodes : uint8 | ||||||
|  | { | ||||||
|  | 	DISCORD_REPLY_NO		UMETA(DisplayName="No"), | ||||||
|  | 	DISCORD_REPLY_YES		UMETA(DisplayName="Yes"), | ||||||
|  | 	DISCORD_REPLY_IGNORE	UMETA(DisplayName="Ignore") | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  | * Valid party privacy values | ||||||
|  | */ | ||||||
|  | UENUM(BlueprintType) | ||||||
|  | enum class EDiscordPartyPrivacy: uint8 | ||||||
|  | { | ||||||
|  |     DISCORD_PARTY_PRIVATE   UMETA(DisplayName="Private"), | ||||||
|  |     DISCORD_PARTY_PUBLIC    UMETA(DisplayName="Public") | ||||||
|  | }; | ||||||
|  |  | ||||||
| DECLARE_LOG_CATEGORY_EXTERN(Discord, Log, All); | DECLARE_LOG_CATEGORY_EXTERN(Discord, Log, All); | ||||||
|  |  | ||||||
| DECLARE_DYNAMIC_MULTICAST_DELEGATE(FDiscordConnected); | DECLARE_DYNAMIC_MULTICAST_DELEGATE_OneParam(FDiscordConnected, const FDiscordUserData&, joinRequest); | ||||||
| DECLARE_DYNAMIC_MULTICAST_DELEGATE_TwoParams(FDiscordDisconnected, int, errorCode, const FString&, errorMessage); | DECLARE_DYNAMIC_MULTICAST_DELEGATE_TwoParams(FDiscordDisconnected, int, errorCode, const FString&, errorMessage); | ||||||
| DECLARE_DYNAMIC_MULTICAST_DELEGATE_TwoParams(FDiscordErrored, int, errorCode, const FString&, errorMessage); | DECLARE_DYNAMIC_MULTICAST_DELEGATE_TwoParams(FDiscordErrored, int, errorCode, const FString&, errorMessage); | ||||||
| DECLARE_DYNAMIC_MULTICAST_DELEGATE_OneParam(FDiscordJoin, const FString&, joinSecret); | DECLARE_DYNAMIC_MULTICAST_DELEGATE_OneParam(FDiscordJoin, const FString&, joinSecret); | ||||||
| DECLARE_DYNAMIC_MULTICAST_DELEGATE_OneParam(FDiscordSpectate, const FString&, spectateSecret); | DECLARE_DYNAMIC_MULTICAST_DELEGATE_OneParam(FDiscordSpectate, const FString&, spectateSecret); | ||||||
| DECLARE_DYNAMIC_MULTICAST_DELEGATE_OneParam(FDiscordJoinRequest, const FDiscordJoinRequestData&, joinRequest); | DECLARE_DYNAMIC_MULTICAST_DELEGATE_OneParam(FDiscordJoinRequest, const FDiscordUserData&, joinRequest); | ||||||
|  |  | ||||||
| // clang-format on | // clang-format on | ||||||
|  |  | ||||||
| @@ -67,6 +87,8 @@ struct FDiscordRichPresence { | |||||||
|     UPROPERTY(BlueprintReadWrite) |     UPROPERTY(BlueprintReadWrite) | ||||||
|     int partyMax; |     int partyMax; | ||||||
|     UPROPERTY(BlueprintReadWrite) |     UPROPERTY(BlueprintReadWrite) | ||||||
|  |     EDiscordPartyPrivacy partyPrivacy; | ||||||
|  |     UPROPERTY(BlueprintReadWrite) | ||||||
|     FString matchSecret; |     FString matchSecret; | ||||||
|     UPROPERTY(BlueprintReadWrite) |     UPROPERTY(BlueprintReadWrite) | ||||||
|     FString joinSecret; |     FString joinSecret; | ||||||
|   | |||||||
| @@ -35,31 +35,34 @@ typedef struct DiscordRichPresence { | |||||||
|     const char* partyId;        /* max 128 bytes */ |     const char* partyId;        /* max 128 bytes */ | ||||||
|     int partySize; |     int partySize; | ||||||
|     int partyMax; |     int partyMax; | ||||||
|  |     int partyPrivacy; | ||||||
|     const char* matchSecret;    /* max 128 bytes */ |     const char* matchSecret;    /* max 128 bytes */ | ||||||
|     const char* joinSecret;     /* max 128 bytes */ |     const char* joinSecret;     /* max 128 bytes */ | ||||||
|     const char* spectateSecret; /* max 128 bytes */ |     const char* spectateSecret; /* max 128 bytes */ | ||||||
|     int8_t instance; |     int8_t instance; | ||||||
| } DiscordRichPresence; | } DiscordRichPresence; | ||||||
|  |  | ||||||
| typedef struct DiscordJoinRequest { | typedef struct DiscordUser { | ||||||
|     const char* userId; |     const char* userId; | ||||||
|     const char* username; |     const char* username; | ||||||
|     const char* discriminator; |     const char* discriminator; | ||||||
|     const char* avatar; |     const char* avatar; | ||||||
| } DiscordJoinRequest; | } DiscordUser; | ||||||
|  |  | ||||||
| typedef struct DiscordEventHandlers { | typedef struct DiscordEventHandlers { | ||||||
|     void (*ready)(void); |     void (*ready)(const DiscordUser* request); | ||||||
|     void (*disconnected)(int errorCode, const char* message); |     void (*disconnected)(int errorCode, const char* message); | ||||||
|     void (*errored)(int errorCode, const char* message); |     void (*errored)(int errorCode, const char* message); | ||||||
|     void (*joinGame)(const char* joinSecret); |     void (*joinGame)(const char* joinSecret); | ||||||
|     void (*spectateGame)(const char* spectateSecret); |     void (*spectateGame)(const char* spectateSecret); | ||||||
|     void (*joinRequest)(const DiscordJoinRequest* request); |     void (*joinRequest)(const DiscordUser* request); | ||||||
| } DiscordEventHandlers; | } DiscordEventHandlers; | ||||||
|  |  | ||||||
| #define DISCORD_REPLY_NO 0 | #define DISCORD_REPLY_NO 0 | ||||||
| #define DISCORD_REPLY_YES 1 | #define DISCORD_REPLY_YES 1 | ||||||
| #define DISCORD_REPLY_IGNORE 2 | #define DISCORD_REPLY_IGNORE 2 | ||||||
|  | #define DISCORD_PARTY_PRIVATE 0 | ||||||
|  | #define DISCORD_PARTY_PUBLIC 1 | ||||||
|  |  | ||||||
| DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | ||||||
|                                        DiscordEventHandlers* handlers, |                                        DiscordEventHandlers* handlers, | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ include_directories(${PROJECT_SOURCE_DIR}/include) | |||||||
|  |  | ||||||
| option(ENABLE_IO_THREAD "Start up a separate I/O thread, otherwise I'd need to call an update function" ON) | option(ENABLE_IO_THREAD "Start up a separate I/O thread, otherwise I'd need to call an update function" ON) | ||||||
| option(USE_STATIC_CRT "Use /MT[d] for dynamic library" OFF) | option(USE_STATIC_CRT "Use /MT[d] for dynamic library" OFF) | ||||||
|  | option(WARNINGS_AS_ERRORS "When enabled, compiles with `-Werror` (on *nix platforms)." OFF) | ||||||
|  |  | ||||||
| set(CMAKE_CXX_STANDARD 14) | set(CMAKE_CXX_STANDARD 14) | ||||||
|  |  | ||||||
| @@ -71,12 +72,23 @@ if(UNIX) | |||||||
|  |  | ||||||
|     add_library(discord-rpc ${BASE_RPC_SRC}) |     add_library(discord-rpc ${BASE_RPC_SRC}) | ||||||
|     target_link_libraries(discord-rpc PUBLIC pthread) |     target_link_libraries(discord-rpc PUBLIC pthread) | ||||||
|  |  | ||||||
|  |     if (APPLE) | ||||||
|  |         target_link_libraries(discord-rpc PRIVATE "-framework AppKit, -mmacosx-version-min=10.10") | ||||||
|  |     endif (APPLE) | ||||||
|  |  | ||||||
|     target_compile_options(discord-rpc PRIVATE |     target_compile_options(discord-rpc PRIVATE | ||||||
|         -g |         -g | ||||||
|         -Wall |         -Wall | ||||||
|         -Wextra |         -Wextra | ||||||
|         -Wpedantic |         -Wpedantic | ||||||
|         -Werror |     ) | ||||||
|  |  | ||||||
|  |     if (${WARNINGS_AS_ERRORS}) | ||||||
|  |       target_compile_options(discord-rpc PRIVATE -Werror) | ||||||
|  |     endif (${WARNINGS_AS_ERRORS}) | ||||||
|  |  | ||||||
|  |     target_compile_options(discord-rpc PRIVATE | ||||||
|         -Wno-unknown-pragmas # pragma push thing doesn't work on clang |         -Wno-unknown-pragmas # pragma push thing doesn't work on clang | ||||||
|         -Wno-old-style-cast # it's fine |         -Wno-old-style-cast # it's fine | ||||||
|         -Wno-c++98-compat # that was almost 2 decades ago |         -Wno-c++98-compat # that was almost 2 decades ago | ||||||
| @@ -133,3 +145,12 @@ install( | |||||||
| 		"../include/discord_register.h" | 		"../include/discord_register.h" | ||||||
|     DESTINATION "include" |     DESTINATION "include" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | if (${BUILD_SHARED_LIBS}) | ||||||
|  |     if(WIN32) | ||||||
|  |         install( | ||||||
|  |             FILES $<TARGET_PDB_FILE:discord-rpc> | ||||||
|  |             DESTINATION "bin" | ||||||
|  |             OPTIONAL) | ||||||
|  |     endif(WIN32) | ||||||
|  | endif(${BUILD_SHARED_LIBS}) | ||||||
|   | |||||||
| @@ -118,5 +118,8 @@ bool BaseConnection::Read(void* data, size_t length) | |||||||
|         } |         } | ||||||
|         Close(); |         Close(); | ||||||
|     } |     } | ||||||
|  |     else if (res == 0) { | ||||||
|  |         Close(); | ||||||
|  |     } | ||||||
|     return res == (int)length; |     return res == (int)length; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -33,13 +33,15 @@ extern "C" DISCORD_EXPORT void Discord_Register(const char* applicationId, const | |||||||
|  |  | ||||||
|     char exePath[1024]; |     char exePath[1024]; | ||||||
|     if (!command || !command[0]) { |     if (!command || !command[0]) { | ||||||
|         if (readlink("/proc/self/exe", exePath, sizeof(exePath)) <= 0) { |         ssize_t size = readlink("/proc/self/exe", exePath, sizeof(exePath)); | ||||||
|  |         if (size <= 0 || size >= (ssize_t)sizeof(exePath)) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |         exePath[size] = '\0'; | ||||||
|         command = exePath; |         command = exePath; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const char* destopFileFormat = "[Desktop Entry]\n" |     const char* desktopFileFormat = "[Desktop Entry]\n" | ||||||
|                                    "Name=Game %s\n" |                                    "Name=Game %s\n" | ||||||
|                                    "Exec=%s %%u\n" // note: it really wants that %u in there |                                    "Exec=%s %%u\n" // note: it really wants that %u in there | ||||||
|                                    "Type=Application\n" |                                    "Type=Application\n" | ||||||
| @@ -48,7 +50,7 @@ extern "C" DISCORD_EXPORT void Discord_Register(const char* applicationId, const | |||||||
|                                    "MimeType=x-scheme-handler/discord-%s;\n"; |                                    "MimeType=x-scheme-handler/discord-%s;\n"; | ||||||
|     char desktopFile[2048]; |     char desktopFile[2048]; | ||||||
|     int fileLen = snprintf( |     int fileLen = snprintf( | ||||||
|       desktopFile, sizeof(desktopFile), destopFileFormat, applicationId, command, applicationId); |       desktopFile, sizeof(desktopFile), desktopFileFormat, applicationId, command, applicationId); | ||||||
|     if (fileLen <= 0) { |     if (fileLen <= 0) { | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
| @@ -91,7 +93,8 @@ extern "C" DISCORD_EXPORT void Discord_Register(const char* applicationId, const | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| extern "C" DISCORD_EXPORT void Discord_RegisterSteamGame(const char* applicationId, const char* steamId) | extern "C" DISCORD_EXPORT void Discord_RegisterSteamGame(const char* applicationId, | ||||||
|  |                                                          const char* steamId) | ||||||
| { | { | ||||||
|     char command[256]; |     char command[256]; | ||||||
|     sprintf(command, "xdg-open steam://rungameid/%s", steamId); |     sprintf(command, "xdg-open steam://rungameid/%s", steamId); | ||||||
|   | |||||||
| @@ -7,7 +7,6 @@ | |||||||
| #define NOIME | #define NOIME | ||||||
| #include <windows.h> | #include <windows.h> | ||||||
| #include <psapi.h> | #include <psapi.h> | ||||||
| #include <cwchar> |  | ||||||
| #include <cstdio> | #include <cstdio> | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -20,19 +19,22 @@ | |||||||
|  * The entire function is rewritten |  * The entire function is rewritten | ||||||
|  */ |  */ | ||||||
| #ifdef __MINGW32__ | #ifdef __MINGW32__ | ||||||
|  | #include <wchar.h> | ||||||
| /// strsafe.h fixes | /// strsafe.h fixes | ||||||
| static HRESULT StringCbPrintfW(LPWSTR pszDest, size_t cbDest, LPCWSTR pszFormat, ...) | static HRESULT StringCbPrintfW(LPWSTR pszDest, size_t cbDest, LPCWSTR pszFormat, ...) | ||||||
| { | { | ||||||
|     HRESULT ret; |     HRESULT ret; | ||||||
|     va_list va; |     va_list va; | ||||||
|     va_start(va, pszFormat); |     va_start(va, pszFormat); | ||||||
|     cbDest /= 2; // Size is divided by 2 to convert from bytes to wide characters - causes segfault othervise |     cbDest /= 2; // Size is divided by 2 to convert from bytes to wide characters - causes segfault | ||||||
|  |                  // othervise | ||||||
|     ret = vsnwprintf(pszDest, cbDest, pszFormat, va); |     ret = vsnwprintf(pszDest, cbDest, pszFormat, va); | ||||||
|     pszDest[cbDest - 1] = 0; // Terminate the string in case a buffer overflow; -1 will be returned |     pszDest[cbDest - 1] = 0; // Terminate the string in case a buffer overflow; -1 will be returned | ||||||
|     va_end(va); |     va_end(va); | ||||||
|     return ret; |     return ret; | ||||||
| } | } | ||||||
| #else | #else | ||||||
|  | #include <cwchar> | ||||||
| #include <strsafe.h> | #include <strsafe.h> | ||||||
| #endif // __MINGW32__ | #endif // __MINGW32__ | ||||||
|  |  | ||||||
| @@ -44,17 +46,24 @@ static HRESULT StringCbPrintfW(LPWSTR pszDest, size_t cbDest, LPCWSTR pszFormat, | |||||||
| #undefine RegSetKeyValueW | #undefine RegSetKeyValueW | ||||||
| #endif | #endif | ||||||
| #define RegSetKeyValueW regset | #define RegSetKeyValueW regset | ||||||
| static LSTATUS regset(HKEY hkey, LPCWSTR subkey, LPCWSTR name, DWORD type, const void *data, DWORD len) | static LSTATUS regset(HKEY hkey, | ||||||
|  |                       LPCWSTR subkey, | ||||||
|  |                       LPCWSTR name, | ||||||
|  |                       DWORD type, | ||||||
|  |                       const void* data, | ||||||
|  |                       DWORD len) | ||||||
| { | { | ||||||
|     HKEY htkey = hkey, hsubkey = nullptr; |     HKEY htkey = hkey, hsubkey = nullptr; | ||||||
|     LSTATUS ret; |     LSTATUS ret; | ||||||
|     if (subkey && subkey[0]) |     if (subkey && subkey[0]) { | ||||||
|     { |         if ((ret = RegCreateKeyExW(hkey, subkey, 0, 0, 0, KEY_ALL_ACCESS, 0, &hsubkey, 0)) != | ||||||
|         if((ret = RegCreateKeyExW(hkey, subkey, 0, 0, 0, KEY_ALL_ACCESS, 0, &hsubkey, 0)) != ERROR_SUCCESS) return ret; |             ERROR_SUCCESS) | ||||||
|  |             return ret; | ||||||
|         htkey = hsubkey; |         htkey = hsubkey; | ||||||
|     } |     } | ||||||
|     ret = RegSetValueExW(htkey, name, 0, type, (const BYTE*)data, len); |     ret = RegSetValueExW(htkey, name, 0, type, (const BYTE*)data, len); | ||||||
|     if (hsubkey && hsubkey != hkey) RegCloseKey(hsubkey); |     if (hsubkey && hsubkey != hkey) | ||||||
|  |         RegCloseKey(hsubkey); | ||||||
|     return ret; |     return ret; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -138,7 +147,8 @@ extern "C" DISCORD_EXPORT void Discord_Register(const char* applicationId, const | |||||||
|     Discord_RegisterW(appId, wcommand); |     Discord_RegisterW(appId, wcommand); | ||||||
| } | } | ||||||
|  |  | ||||||
| extern "C" DISCORD_EXPORT void Discord_RegisterSteamGame(const char* applicationId, const char* steamId) | extern "C" DISCORD_EXPORT void Discord_RegisterSteamGame(const char* applicationId, | ||||||
|  |                                                          const char* steamId) | ||||||
| { | { | ||||||
|     wchar_t appId[32]; |     wchar_t appId[32]; | ||||||
|     MultiByteToWideChar(CP_UTF8, 0, applicationId, -1, appId, 32); |     MultiByteToWideChar(CP_UTF8, 0, applicationId, -1, appId, 32); | ||||||
|   | |||||||
| @@ -32,7 +32,7 @@ struct QueuedMessage { | |||||||
|     } |     } | ||||||
| }; | }; | ||||||
|  |  | ||||||
| struct JoinRequest { | struct User { | ||||||
|     // snowflake (64bit int), turned into a ascii decimal string, at most 20 chars +1 null |     // snowflake (64bit int), turned into a ascii decimal string, at most 20 chars +1 null | ||||||
|     // terminator = 21 |     // terminator = 21 | ||||||
|     char userId[32]; |     char userId[32]; | ||||||
| @@ -47,12 +47,14 @@ struct JoinRequest { | |||||||
| }; | }; | ||||||
|  |  | ||||||
| static RpcConnection* Connection{nullptr}; | static RpcConnection* Connection{nullptr}; | ||||||
|  | static DiscordEventHandlers QueuedHandlers{}; | ||||||
| static DiscordEventHandlers Handlers{}; | static DiscordEventHandlers Handlers{}; | ||||||
| static std::atomic_bool WasJustConnected{false}; | static std::atomic_bool WasJustConnected{false}; | ||||||
| static std::atomic_bool WasJustDisconnected{false}; | static std::atomic_bool WasJustDisconnected{false}; | ||||||
| static std::atomic_bool GotErrorMessage{false}; | static std::atomic_bool GotErrorMessage{false}; | ||||||
| static std::atomic_bool WasJoinGame{false}; | static std::atomic_bool WasJoinGame{false}; | ||||||
| static std::atomic_bool WasSpectateGame{false}; | static std::atomic_bool WasSpectateGame{false}; | ||||||
|  | static std::atomic_bool UpdatePresence{false}; | ||||||
| static char JoinGameSecret[256]; | static char JoinGameSecret[256]; | ||||||
| static char SpectateGameSecret[256]; | static char SpectateGameSecret[256]; | ||||||
| static int LastErrorCode{0}; | static int LastErrorCode{0}; | ||||||
| @@ -63,7 +65,8 @@ static std::mutex PresenceMutex; | |||||||
| static std::mutex HandlerMutex; | static std::mutex HandlerMutex; | ||||||
| static QueuedMessage QueuedPresence{}; | static QueuedMessage QueuedPresence{}; | ||||||
| static MsgQueue<QueuedMessage, MessageQueueSize> SendQueue; | static MsgQueue<QueuedMessage, MessageQueueSize> SendQueue; | ||||||
| static MsgQueue<JoinRequest, JoinQueueSize> JoinAskQueue; | static MsgQueue<User, JoinQueueSize> JoinAskQueue; | ||||||
|  | static User connectedUser; | ||||||
|  |  | ||||||
| // We want to auto connect, and retry on failure, but not as fast as possible. This does expoential | // We want to auto connect, and retry on failure, but not as fast as possible. This does expoential | ||||||
| // backoff from 0.5 seconds to 1 minute | // backoff from 0.5 seconds to 1 minute | ||||||
| @@ -87,10 +90,11 @@ public: | |||||||
|         keepRunning.store(true); |         keepRunning.store(true); | ||||||
|         ioThread = std::thread([&]() { |         ioThread = std::thread([&]() { | ||||||
|             const std::chrono::duration<int64_t, std::milli> maxWait{500LL}; |             const std::chrono::duration<int64_t, std::milli> maxWait{500LL}; | ||||||
|             while (keepRunning.load()) { |  | ||||||
|             Discord_UpdateConnection(); |             Discord_UpdateConnection(); | ||||||
|  |             while (keepRunning.load()) { | ||||||
|                 std::unique_lock<std::mutex> lock(waitForIOMutex); |                 std::unique_lock<std::mutex> lock(waitForIOMutex); | ||||||
|                 waitForIOActivity.wait_for(lock, maxWait); |                 waitForIOActivity.wait_for(lock, maxWait); | ||||||
|  |                 Discord_UpdateConnection(); | ||||||
|             } |             } | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
| @@ -116,7 +120,7 @@ public: | |||||||
|     void Notify() {} |     void Notify() {} | ||||||
| }; | }; | ||||||
| #endif // DISCORD_DISABLE_IO_THREAD | #endif // DISCORD_DISABLE_IO_THREAD | ||||||
| static IoThreadHolder IoThread; | static IoThreadHolder* IoThread{nullptr}; | ||||||
|  |  | ||||||
| static void UpdateReconnectTime() | static void UpdateReconnectTime() | ||||||
| { | { | ||||||
| @@ -211,17 +215,17 @@ static void Discord_UpdateConnection(void) | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         // writes |         // writes | ||||||
|         if (QueuedPresence.length) { |         if (UpdatePresence.exchange(false) && QueuedPresence.length) { | ||||||
|             QueuedMessage local; |             QueuedMessage local; | ||||||
|             { |             { | ||||||
|                 std::lock_guard<std::mutex> guard(PresenceMutex); |                 std::lock_guard<std::mutex> guard(PresenceMutex); | ||||||
|                 local.Copy(QueuedPresence); |                 local.Copy(QueuedPresence); | ||||||
|                 QueuedPresence.length = 0; |  | ||||||
|             } |             } | ||||||
|             if (!Connection->Write(local.buffer, local.length)) { |             if (!Connection->Write(local.buffer, local.length)) { | ||||||
|                 // if we fail to send, requeue |                 // if we fail to send, requeue | ||||||
|                 std::lock_guard<std::mutex> guard(PresenceMutex); |                 std::lock_guard<std::mutex> guard(PresenceMutex); | ||||||
|                 QueuedPresence.Copy(local); |                 QueuedPresence.Copy(local); | ||||||
|  |                 UpdatePresence.exchange(true); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -235,7 +239,9 @@ static void Discord_UpdateConnection(void) | |||||||
|  |  | ||||||
| static void SignalIOActivity() | static void SignalIOActivity() | ||||||
| { | { | ||||||
|     IoThread.Notify(); |     if (IoThread != nullptr) { | ||||||
|  |         IoThread->Notify(); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| static bool RegisterForEvent(const char* evtName) | static bool RegisterForEvent(const char* evtName) | ||||||
| @@ -269,6 +275,11 @@ extern "C" DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | |||||||
|                                                   int autoRegister, |                                                   int autoRegister, | ||||||
|                                                   const char* optionalSteamId) |                                                   const char* optionalSteamId) | ||||||
| { | { | ||||||
|  |     IoThread = new (std::nothrow) IoThreadHolder(); | ||||||
|  |     if (IoThread == nullptr) { | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     if (autoRegister) { |     if (autoRegister) { | ||||||
|         if (optionalSteamId && optionalSteamId[0]) { |         if (optionalSteamId && optionalSteamId[0]) { | ||||||
|             Discord_RegisterSteamGame(applicationId, optionalSteamId); |             Discord_RegisterSteamGame(applicationId, optionalSteamId); | ||||||
| @@ -282,12 +293,15 @@ extern "C" DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | |||||||
|  |  | ||||||
|     { |     { | ||||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); |         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||||
|  |  | ||||||
|         if (handlers) { |         if (handlers) { | ||||||
|             Handlers = *handlers; |             QueuedHandlers = *handlers; | ||||||
|         } |         } | ||||||
|         else { |         else { | ||||||
|             Handlers = {}; |             QueuedHandlers = {}; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         Handlers = {}; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (Connection) { |     if (Connection) { | ||||||
| @@ -295,8 +309,31 @@ extern "C" DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     Connection = RpcConnection::Create(applicationId); |     Connection = RpcConnection::Create(applicationId); | ||||||
|     Connection->onConnect = []() { |     Connection->onConnect = [](JsonDocument& readyMessage) { | ||||||
|         Discord_UpdateHandlers(&Handlers); |         Discord_UpdateHandlers(&QueuedHandlers); | ||||||
|  |         if (QueuedPresence.length > 0) { | ||||||
|  |             UpdatePresence.exchange(true); | ||||||
|  |             SignalIOActivity(); | ||||||
|  |         } | ||||||
|  |         auto data = GetObjMember(&readyMessage, "data"); | ||||||
|  |         auto user = GetObjMember(data, "user"); | ||||||
|  |         auto userId = GetStrMember(user, "id"); | ||||||
|  |         auto username = GetStrMember(user, "username"); | ||||||
|  |         auto avatar = GetStrMember(user, "avatar"); | ||||||
|  |         if (userId && username) { | ||||||
|  |             StringCopy(connectedUser.userId, userId); | ||||||
|  |             StringCopy(connectedUser.username, username); | ||||||
|  |             auto discriminator = GetStrMember(user, "discriminator"); | ||||||
|  |             if (discriminator) { | ||||||
|  |                 StringCopy(connectedUser.discriminator, discriminator); | ||||||
|  |             } | ||||||
|  |             if (avatar) { | ||||||
|  |                 StringCopy(connectedUser.avatar, avatar); | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 connectedUser.avatar[0] = 0; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|         WasJustConnected.exchange(true); |         WasJustConnected.exchange(true); | ||||||
|         ReconnectTimeMs.reset(); |         ReconnectTimeMs.reset(); | ||||||
|     }; |     }; | ||||||
| @@ -307,7 +344,7 @@ extern "C" DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | |||||||
|         UpdateReconnectTime(); |         UpdateReconnectTime(); | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     IoThread.Start(); |     IoThread->Start(); | ||||||
| } | } | ||||||
|  |  | ||||||
| extern "C" DISCORD_EXPORT void Discord_Shutdown(void) | extern "C" DISCORD_EXPORT void Discord_Shutdown(void) | ||||||
| @@ -318,7 +355,14 @@ extern "C" DISCORD_EXPORT void Discord_Shutdown(void) | |||||||
|     Connection->onConnect = nullptr; |     Connection->onConnect = nullptr; | ||||||
|     Connection->onDisconnect = nullptr; |     Connection->onDisconnect = nullptr; | ||||||
|     Handlers = {}; |     Handlers = {}; | ||||||
|     IoThread.Stop(); |     QueuedPresence.length = 0; | ||||||
|  |     UpdatePresence.exchange(false); | ||||||
|  |     if (IoThread != nullptr) { | ||||||
|  |         IoThread->Stop(); | ||||||
|  |         delete IoThread; | ||||||
|  |         IoThread = nullptr; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     RpcConnection::Destroy(Connection); |     RpcConnection::Destroy(Connection); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -328,6 +372,7 @@ extern "C" DISCORD_EXPORT void Discord_UpdatePresence(const DiscordRichPresence* | |||||||
|         std::lock_guard<std::mutex> guard(PresenceMutex); |         std::lock_guard<std::mutex> guard(PresenceMutex); | ||||||
|         QueuedPresence.length = JsonWriteRichPresenceObj( |         QueuedPresence.length = JsonWriteRichPresenceObj( | ||||||
|           QueuedPresence.buffer, sizeof(QueuedPresence.buffer), Nonce++, Pid, presence); |           QueuedPresence.buffer, sizeof(QueuedPresence.buffer), Nonce++, Pid, presence); | ||||||
|  |         UpdatePresence.exchange(true); | ||||||
|     } |     } | ||||||
|     SignalIOActivity(); |     SignalIOActivity(); | ||||||
| } | } | ||||||
| @@ -376,7 +421,11 @@ extern "C" DISCORD_EXPORT void Discord_RunCallbacks(void) | |||||||
|     if (WasJustConnected.exchange(false)) { |     if (WasJustConnected.exchange(false)) { | ||||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); |         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||||
|         if (Handlers.ready) { |         if (Handlers.ready) { | ||||||
|             Handlers.ready(); |             DiscordUser du{connectedUser.userId, | ||||||
|  |                            connectedUser.username, | ||||||
|  |                            connectedUser.discriminator, | ||||||
|  |                            connectedUser.avatar}; | ||||||
|  |             Handlers.ready(&du); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -411,8 +460,8 @@ extern "C" DISCORD_EXPORT void Discord_RunCallbacks(void) | |||||||
|         { |         { | ||||||
|             std::lock_guard<std::mutex> guard(HandlerMutex); |             std::lock_guard<std::mutex> guard(HandlerMutex); | ||||||
|             if (Handlers.joinRequest) { |             if (Handlers.joinRequest) { | ||||||
|                 DiscordJoinRequest djr{req->userId, req->username, req->discriminator, req->avatar}; |                 DiscordUser du{req->userId, req->username, req->discriminator, req->avatar}; | ||||||
|                 Handlers.joinRequest(&djr); |                 Handlers.joinRequest(&du); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         JoinAskQueue.CommitSend(); |         JoinAskQueue.CommitSend(); | ||||||
| @@ -430,7 +479,6 @@ extern "C" DISCORD_EXPORT void Discord_RunCallbacks(void) | |||||||
| extern "C" DISCORD_EXPORT void Discord_UpdateHandlers(DiscordEventHandlers* newHandlers) | extern "C" DISCORD_EXPORT void Discord_UpdateHandlers(DiscordEventHandlers* newHandlers) | ||||||
| { | { | ||||||
|     if (newHandlers) { |     if (newHandlers) { | ||||||
|  |  | ||||||
| #define HANDLE_EVENT_REGISTRATION(handler_name, event)              \ | #define HANDLE_EVENT_REGISTRATION(handler_name, event)              \ | ||||||
|     if (!Handlers.handler_name && newHandlers->handler_name) {      \ |     if (!Handlers.handler_name && newHandlers->handler_name) {      \ | ||||||
|         RegisterForEvent(event);                                    \ |         RegisterForEvent(event);                                    \ | ||||||
| @@ -448,8 +496,7 @@ extern "C" DISCORD_EXPORT void Discord_UpdateHandlers(DiscordEventHandlers* newH | |||||||
|  |  | ||||||
|         Handlers = *newHandlers; |         Handlers = *newHandlers; | ||||||
|     } |     } | ||||||
|     else |     else { | ||||||
|     { |  | ||||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); |         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||||
|         Handlers = {}; |         Handlers = {}; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ | |||||||
|  |  | ||||||
| template <typename ElementType, size_t QueueSize> | template <typename ElementType, size_t QueueSize> | ||||||
| class MsgQueue { | class MsgQueue { | ||||||
|     ElementType queue_[QueueSize]{}; |     ElementType queue_[QueueSize]; | ||||||
|     std::atomic_uint nextAdd_{0}; |     std::atomic_uint nextAdd_{0}; | ||||||
|     std::atomic_uint nextSend_{0}; |     std::atomic_uint nextSend_{0}; | ||||||
|     std::atomic_uint pendingSends_{0}; |     std::atomic_uint pendingSends_{0}; | ||||||
|   | |||||||
| @@ -26,13 +26,9 @@ void RpcConnection::Open() | |||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (state == State::Disconnected) { |     if (state == State::Disconnected && !connection->Open()) { | ||||||
|         if (connection->Open()) { |  | ||||||
|         } |  | ||||||
|         else { |  | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (state == State::SentHandshake) { |     if (state == State::SentHandshake) { | ||||||
|         JsonDocument message; |         JsonDocument message; | ||||||
| @@ -42,7 +38,7 @@ void RpcConnection::Open() | |||||||
|             if (cmd && evt && !strcmp(cmd, "DISPATCH") && !strcmp(evt, "READY")) { |             if (cmd && evt && !strcmp(cmd, "DISPATCH") && !strcmp(evt, "READY")) { | ||||||
|                 state = State::Connected; |                 state = State::Connected; | ||||||
|                 if (onConnect) { |                 if (onConnect) { | ||||||
|                     onConnect(); |                     onConnect(message); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -40,7 +40,7 @@ struct RpcConnection { | |||||||
|  |  | ||||||
|     BaseConnection* connection{nullptr}; |     BaseConnection* connection{nullptr}; | ||||||
|     State state{State::Disconnected}; |     State state{State::Disconnected}; | ||||||
|     void (*onConnect)(){nullptr}; |     void (*onConnect)(JsonDocument& message){nullptr}; | ||||||
|     void (*onDisconnect)(int errorCode, const char* message){nullptr}; |     void (*onDisconnect)(int errorCode, const char* message){nullptr}; | ||||||
|     char appId[64]{}; |     char appId[64]{}; | ||||||
|     int lastErrorCode{0}; |     int lastErrorCode{0}; | ||||||
|   | |||||||
| @@ -102,8 +102,7 @@ size_t JsonWriteRichPresenceObj(char* dest, | |||||||
|             WriteKey(writer, "pid"); |             WriteKey(writer, "pid"); | ||||||
|             writer.Int(pid); |             writer.Int(pid); | ||||||
|  |  | ||||||
|             if (presence != nullptr) |             if (presence != nullptr) { | ||||||
|             { |  | ||||||
|                 WriteObject activity(writer, "activity"); |                 WriteObject activity(writer, "activity"); | ||||||
|  |  | ||||||
|                 WriteOptionalString(writer, "state", presence->state); |                 WriteOptionalString(writer, "state", presence->state); | ||||||
| @@ -135,7 +134,7 @@ size_t JsonWriteRichPresenceObj(char* dest, | |||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 if ((presence->partyId && presence->partyId[0]) || presence->partySize || |                 if ((presence->partyId && presence->partyId[0]) || presence->partySize || | ||||||
|                     presence->partyMax) { |                     presence->partyMax || presence->partyPrivacy) { | ||||||
|                     WriteObject party(writer, "party"); |                     WriteObject party(writer, "party"); | ||||||
|                     WriteOptionalString(writer, "id", presence->partyId); |                     WriteOptionalString(writer, "id", presence->partyId); | ||||||
|                     if (presence->partySize && presence->partyMax) { |                     if (presence->partySize && presence->partyMax) { | ||||||
| @@ -143,6 +142,11 @@ size_t JsonWriteRichPresenceObj(char* dest, | |||||||
|                         writer.Int(presence->partySize); |                         writer.Int(presence->partySize); | ||||||
|                         writer.Int(presence->partyMax); |                         writer.Int(presence->partyMax); | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|  |                     if (presence->partyPrivacy) { | ||||||
|  |                         WriteKey(writer, "privacy"); | ||||||
|  |                         writer.Int(presence->partyPrivacy); | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 if ((presence->matchSecret && presence->matchSecret[0]) || |                 if ((presence->matchSecret && presence->matchSecret[0]) || | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user