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: | ||||
|     - mkdir 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 | ||||
|   | ||||
							
								
								
									
										95
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										95
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,5 +1,11 @@ | ||||
| # 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. | ||||
|  | ||||
| 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. | ||||
|  | ||||
| ### 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 | ||||
|  | ||||
| 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 | ||||
|  | ||||
| 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: | ||||
|  | ||||
| ```sh | ||||
|     cd <path to discord-rpc> | ||||
|     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 --build . --config Release --target install | ||||
| ``` | ||||
|  | ||||
| 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`. | ||||
| @@ -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: | ||||
|  | ||||
| | 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. | ||||
| | `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 | ||||
| | ---------------------------------------------------------------------------------------- | ------- | ----------------------------------------------------------------------------------------------------------------------------------------------------- | | ||||
| | `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) | | ||||
| | [`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 | ||||
|  | ||||
| Why do we have three of these? Three times the fun! | ||||
|  | ||||
| | CI                   | badge                                                                                                                                            | | ||||
| |----|-------| | ||||
| | TravisCI | [](https://travis-ci.org/discordapp/discord-rpc) | ||||
| | AppVeyor | [](https://ci.appveyor.com/project/crmarsh/discord-rpc) | ||||
| | Buildkite (internal) | [](https://buildkite.com/discord/discord-rpc) | ||||
| | -------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------ | | ||||
| | TravisCI             | [](https://travis-ci.org/discordapp/discord-rpc)                  | | ||||
| | AppVeyor             | [](https://ci.appveyor.com/project/crmarsh/discord-rpc)    | | ||||
| | Buildkite (internal) | [](https://buildkite.com/discord/discord-rpc) | | ||||
|  | ||||
| ## Sample: send-presence | ||||
|  | ||||
| @@ -74,10 +145,14 @@ Below is a table of unofficial, community-developed wrappers for and implementat | ||||
| ###### Rich Presence Wrappers and Implementations | ||||
|  | ||||
| | 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                              | | ||||
| | [Discord-IPC](https://github.com/jagrosh/DiscordIPC)                      | Java                              | | ||||
| | [Discord Rich Presence](https://npmjs.org/discord-rich-presence)          | JavaScript                        | | ||||
| | [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)        | | ||||
|   | ||||
							
								
								
									
										59
									
								
								build.py
									
									
									
									
									
								
							
							
						
						
									
										59
									
								
								build.py
									
									
									
									
									
								
							| @@ -30,7 +30,7 @@ INSTALL_ROOT = os.path.join(SCRIPT_PATH, 'builds', 'install') | ||||
| def get_signtool(): | ||||
|     """ get path to code signing tool """ | ||||
|     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') | ||||
|     elif PLATFORM == 'osx': | ||||
|         return '/usr/bin/codesign' | ||||
| @@ -70,14 +70,7 @@ def cli(ctx, clean): | ||||
| @click.pass_context | ||||
| def unity(ctx): | ||||
|     """ build just dynamic libs for use in unity project """ | ||||
|     ctx.invoke( | ||||
|         libs, | ||||
|         clean=False, | ||||
|         static=False, | ||||
|         shared=True, | ||||
|         skip_formatter=True, | ||||
|         just_release=True | ||||
|     ) | ||||
|     ctx.invoke(libs, clean=False, static=False, shared=True, skip_formatter=True, just_release=True) | ||||
|     BUILDS = [] | ||||
|  | ||||
|     click.echo('--- Copying libs and header into unity example') | ||||
| @@ -85,11 +78,11 @@ def unity(ctx): | ||||
|  | ||||
|     if sys.platform.startswith('win'): | ||||
|         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') | ||||
|         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') | ||||
|         BUILDS.append({BUILD_32_BASE_PATH: UNITY_32_DLL_PATH}) | ||||
|  | ||||
| @@ -97,7 +90,8 @@ def unity(ctx): | ||||
|         LIBRARY_NAME = 'discord-rpc.bundle' | ||||
|         BUILD_BASE_PATH = os.path.join(SCRIPT_PATH, 'builds', 'osx-dynamic', 'src') | ||||
|         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}) | ||||
|  | ||||
| @@ -122,14 +116,7 @@ def unity(ctx): | ||||
| @click.pass_context | ||||
| def unreal(ctx): | ||||
|     """ build libs and copy them into the unreal project """ | ||||
|     ctx.invoke( | ||||
|         libs, | ||||
|         clean=False, | ||||
|         static=False, | ||||
|         shared=True, | ||||
|         skip_formatter=True, | ||||
|         just_release=True | ||||
|     ) | ||||
|     ctx.invoke(libs, clean=False, static=False, shared=True, skip_formatter=True, just_release=True) | ||||
|     BUILDS = [] | ||||
|  | ||||
|     click.echo('--- Copying libs and header into unreal example') | ||||
| @@ -140,11 +127,11 @@ def unreal(ctx): | ||||
|  | ||||
|     if sys.platform.startswith('win'): | ||||
|         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') | ||||
|         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') | ||||
|         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(install_path) | ||||
|     with cd(build_path): | ||||
|         initial_cmake = [ | ||||
|             'cmake', | ||||
|             SCRIPT_PATH, | ||||
|             '-DCMAKE_INSTALL_PREFIX=%s' % os.path.join('..', 'install', build_name) | ||||
|         ] | ||||
|         initial_cmake = ['cmake', SCRIPT_PATH, '-DCMAKE_INSTALL_PREFIX=%s' % os.path.join('..', 'install', build_name)] | ||||
|         if generator: | ||||
|             initial_cmake.extend(['-G', generator]) | ||||
|         for key in options: | ||||
| @@ -194,7 +177,7 @@ def build_lib(build_name, generator, options, just_release): | ||||
|         subprocess.check_call(initial_cmake) | ||||
|         if not just_release: | ||||
|             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() | ||||
| @@ -224,22 +207,28 @@ def sign(): | ||||
|         sign_command_base = [ | ||||
|             tool, | ||||
|             'sign', | ||||
|             '/n', 'Discord Inc.', | ||||
|             '/n', | ||||
|             'Discord Inc.', | ||||
|             '/a', | ||||
|             '/tr', 'http://timestamp.digicert.com/rfc3161', | ||||
|             '/tr', | ||||
|             'http://timestamp.digicert.com/rfc3161', | ||||
|             '/as', | ||||
|             '/td', 'sha256', | ||||
|             '/fd', 'sha256', | ||||
|             '/td', | ||||
|             'sha256', | ||||
|             '/fd', | ||||
|             'sha256', | ||||
|         ] | ||||
|     elif PLATFORM == 'osx': | ||||
|         signable_extensions.add('.dylib') | ||||
|         sign_command_base = [ | ||||
|             tool, | ||||
|             '--keychain', os.path.expanduser('~/Library/Keychains/login.keychain'), | ||||
|             '--keychain', | ||||
|             os.path.expanduser('~/Library/Keychains/login.keychain'), | ||||
|             '-vvvv', | ||||
|             '--deep', | ||||
|             '--force', | ||||
|             '--sign', 'Developer ID Application: Hammer & Chisel Inc. (53Q6R32WPB)', | ||||
|             '--sign', | ||||
|             'Developer ID Application: Hammer & Chisel Inc. (53Q6R32WPB)', | ||||
|         ] | ||||
|     else: | ||||
|         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: | ||||
|         just_release = True | ||||
|         static_options['WARNINGS_AS_ERRORS'] = True | ||||
|         dynamic_options['WARNINGS_AS_ERRORS'] = True | ||||
|  | ||||
|     if PLATFORM == 'win': | ||||
|         generator32 = 'Visual Studio 14 2015' | ||||
|   | ||||
| @@ -7,16 +7,15 @@ public class DiscordJoinEvent : UnityEngine.Events.UnityEvent<string> { } | ||||
| public class DiscordSpectateEvent : UnityEngine.Events.UnityEvent<string> { } | ||||
|  | ||||
| [System.Serializable] | ||||
| public class DiscordJoinRequestEvent : UnityEngine.Events.UnityEvent<DiscordRpc.JoinRequest> { } | ||||
| public class DiscordJoinRequestEvent : UnityEngine.Events.UnityEvent<DiscordRpc.DiscordUser> { } | ||||
|  | ||||
| public class DiscordController : MonoBehaviour | ||||
| { | ||||
|     public DiscordRpc.RichPresence presence = new DiscordRpc.RichPresence(); | ||||
|     public string applicationId; | ||||
|     public string optionalSteamId; | ||||
|     public int callbackCalls; | ||||
|     public int clickCounter; | ||||
|     public DiscordRpc.JoinRequest joinRequest; | ||||
|     public DiscordRpc.DiscordUser joinRequest; | ||||
|     public UnityEngine.Events.UnityEvent onConnect; | ||||
|     public UnityEngine.Events.UnityEvent onDisconnect; | ||||
|     public UnityEngine.Events.UnityEvent hasResponded; | ||||
| @@ -32,6 +31,11 @@ public class DiscordController : MonoBehaviour | ||||
|         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); | ||||
|     } | ||||
| @@ -50,43 +54,37 @@ public class DiscordController : MonoBehaviour | ||||
|         hasResponded.Invoke(); | ||||
|     } | ||||
|  | ||||
|     public void ReadyCallback() | ||||
|     public void ReadyCallback(ref DiscordRpc.DiscordUser connectedUser) | ||||
|     { | ||||
|         ++callbackCalls; | ||||
|         Debug.Log("Discord: ready"); | ||||
|         Debug.Log(string.Format("Discord: connected to {0}#{1}: {2}", connectedUser.username, connectedUser.discriminator, connectedUser.userId)); | ||||
|         onConnect.Invoke(); | ||||
|     } | ||||
|  | ||||
|     public void DisconnectedCallback(int errorCode, string message) | ||||
|     { | ||||
|         ++callbackCalls; | ||||
|         Debug.Log(string.Format("Discord: disconnect {0}: {1}", errorCode, message)); | ||||
|         onDisconnect.Invoke(); | ||||
|     } | ||||
|  | ||||
|     public void ErrorCallback(int errorCode, string message) | ||||
|     { | ||||
|         ++callbackCalls; | ||||
|         Debug.Log(string.Format("Discord: error {0}: {1}", errorCode, message)); | ||||
|     } | ||||
|  | ||||
|     public void JoinCallback(string secret) | ||||
|     { | ||||
|         ++callbackCalls; | ||||
|         Debug.Log(string.Format("Discord: join ({0})", secret)); | ||||
|         onJoin.Invoke(secret); | ||||
|     } | ||||
|  | ||||
|     public void SpectateCallback(string secret) | ||||
|     { | ||||
|         ++callbackCalls; | ||||
|         Debug.Log(string.Format("Discord: spectate ({0})", 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)); | ||||
|         joinRequest = request; | ||||
|         onJoinRequest.Invoke(request); | ||||
| @@ -104,10 +102,8 @@ public class DiscordController : MonoBehaviour | ||||
|     void OnEnable() | ||||
|     { | ||||
|         Debug.Log("Discord: init"); | ||||
|         callbackCalls = 0; | ||||
|  | ||||
|         handlers = new DiscordRpc.EventHandlers(); | ||||
|         handlers.readyCallback = ReadyCallback; | ||||
|         handlers.readyCallback += ReadyCallback; | ||||
|         handlers.disconnectedCallback += DisconnectedCallback; | ||||
|         handlers.errorCallback += ErrorCallback; | ||||
|         handlers.joinCallback += JoinCallback; | ||||
|   | ||||
| @@ -2,35 +2,44 @@ | ||||
| using System.Collections.Generic; | ||||
| using System.Runtime.InteropServices; | ||||
| using System.Text; | ||||
| using AOT; | ||||
|  | ||||
| public class DiscordRpc | ||||
| { | ||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] | ||||
|     public delegate void ReadyCallback(); | ||||
|     [MonoPInvokeCallback(typeof(OnReadyInfo))] | ||||
|     public static void ReadyCallback(ref DiscordUser connectedUser) { Callbacks.readyCallback(ref connectedUser); } | ||||
|     public delegate void OnReadyInfo(ref DiscordUser connectedUser); | ||||
|  | ||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] | ||||
|     public delegate void DisconnectedCallback(int errorCode, string message); | ||||
|     [MonoPInvokeCallback(typeof(OnDisconnectedInfo))] | ||||
|     public static void DisconnectedCallback(int errorCode, string message) { Callbacks.disconnectedCallback(errorCode, message); } | ||||
|     public delegate void OnDisconnectedInfo(int errorCode, string message); | ||||
|  | ||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] | ||||
|     public delegate void ErrorCallback(int errorCode, string message); | ||||
|     [MonoPInvokeCallback(typeof(OnErrorInfo))] | ||||
|     public static void ErrorCallback(int errorCode, string message) { Callbacks.errorCallback(errorCode, message); } | ||||
|     public delegate void OnErrorInfo(int errorCode, string message); | ||||
|  | ||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] | ||||
|     public delegate void JoinCallback(string secret); | ||||
|     [MonoPInvokeCallback(typeof(OnJoinInfo))] | ||||
|     public static void JoinCallback(string secret) { Callbacks.joinCallback(secret); } | ||||
|     public delegate void OnJoinInfo(string secret); | ||||
|  | ||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] | ||||
|     public delegate void SpectateCallback(string secret); | ||||
|     [MonoPInvokeCallback(typeof(OnSpectateInfo))] | ||||
|     public static void SpectateCallback(string secret) { Callbacks.spectateCallback(secret); } | ||||
|     public delegate void OnSpectateInfo(string secret); | ||||
|  | ||||
|     [UnmanagedFunctionPointer(CallingConvention.Cdecl)] | ||||
|     public delegate void RequestCallback(ref JoinRequest request); | ||||
|     [MonoPInvokeCallback(typeof(OnRequestInfo))] | ||||
|     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 ReadyCallback readyCallback; | ||||
|         public DisconnectedCallback disconnectedCallback; | ||||
|         public ErrorCallback errorCallback; | ||||
|         public JoinCallback joinCallback; | ||||
|         public SpectateCallback spectateCallback; | ||||
|         public RequestCallback requestCallback; | ||||
|         public OnReadyInfo readyCallback; | ||||
|         public OnDisconnectedInfo disconnectedCallback; | ||||
|         public OnErrorInfo errorCallback; | ||||
|         public OnJoinInfo joinCallback; | ||||
|         public OnSpectateInfo spectateCallback; | ||||
|         public OnRequestInfo requestCallback; | ||||
|     } | ||||
|  | ||||
|     [Serializable, StructLayout(LayoutKind.Sequential)] | ||||
| @@ -47,6 +56,7 @@ public class DiscordRpc | ||||
|         public IntPtr partyId; /* max 128 bytes */ | ||||
|         public int partySize; | ||||
|         public int partyMax; | ||||
|         public int partyPrivacy; | ||||
|         public IntPtr matchSecret; /* max 128 bytes */ | ||||
|         public IntPtr joinSecret; /* max 128 bytes */ | ||||
|         public IntPtr spectateSecret; /* max 128 bytes */ | ||||
| @@ -54,7 +64,7 @@ public class DiscordRpc | ||||
|     } | ||||
|  | ||||
|     [Serializable] | ||||
|     public struct JoinRequest | ||||
|     public struct DiscordUser | ||||
|     { | ||||
|         public string userId; | ||||
|         public string username; | ||||
| @@ -69,8 +79,29 @@ public class DiscordRpc | ||||
|         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)] | ||||
|     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)] | ||||
|     public static extern void Shutdown(); | ||||
| @@ -113,6 +144,7 @@ public class DiscordRpc | ||||
|         public string partyId; /* max 128 bytes */ | ||||
|         public int partySize; | ||||
|         public int partyMax; | ||||
|         public PartyPrivacy partyPrivacy; | ||||
|         public string matchSecret; /* max 128 bytes */ | ||||
|         public string joinSecret; /* max 128 bytes */ | ||||
|         public string spectateSecret; /* max 128 bytes */ | ||||
| @@ -129,20 +161,21 @@ public class DiscordRpc | ||||
|                 FreeMem(); | ||||
|             } | ||||
|  | ||||
|             _presence.state = StrToPtr(state, 128); | ||||
|             _presence.details = StrToPtr(details, 128); | ||||
|             _presence.state = StrToPtr(state); | ||||
|             _presence.details = StrToPtr(details); | ||||
|             _presence.startTimestamp = startTimestamp; | ||||
|             _presence.endTimestamp = endTimestamp; | ||||
|             _presence.largeImageKey = StrToPtr(largeImageKey, 32); | ||||
|             _presence.largeImageText = StrToPtr(largeImageText, 128); | ||||
|             _presence.smallImageKey = StrToPtr(smallImageKey, 32); | ||||
|             _presence.smallImageText = StrToPtr(smallImageText, 128); | ||||
|             _presence.partyId = StrToPtr(partyId, 128); | ||||
|             _presence.largeImageKey = StrToPtr(largeImageKey); | ||||
|             _presence.largeImageText = StrToPtr(largeImageText); | ||||
|             _presence.smallImageKey = StrToPtr(smallImageKey); | ||||
|             _presence.smallImageText = StrToPtr(smallImageText); | ||||
|             _presence.partyId = StrToPtr(partyId); | ||||
|             _presence.partySize = partySize; | ||||
|             _presence.partyMax = partyMax; | ||||
|             _presence.matchSecret = StrToPtr(matchSecret, 128); | ||||
|             _presence.joinSecret = StrToPtr(joinSecret, 128); | ||||
|             _presence.spectateSecret = StrToPtr(spectateSecret, 128); | ||||
|             _presence.partyPrivacy = (int)partyPrivacy; | ||||
|             _presence.matchSecret = StrToPtr(matchSecret); | ||||
|             _presence.joinSecret = StrToPtr(joinSecret); | ||||
|             _presence.spectateSecret = StrToPtr(spectateSecret); | ||||
|             _presence.instance = instance; | ||||
|  | ||||
|             return _presence; | ||||
| @@ -152,16 +185,18 @@ public class DiscordRpc | ||||
|         /// Returns a pointer to a representation of the given string with a size of maxbytes | ||||
|         /// </summary> | ||||
|         /// <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> | ||||
|         private IntPtr StrToPtr(string input, int maxbytes) | ||||
|         private IntPtr StrToPtr(string input) | ||||
|         { | ||||
|             if (string.IsNullOrEmpty(input)) return IntPtr.Zero; | ||||
|             var convstr = StrClampBytes(input, maxbytes); | ||||
|             var convbytecnt = Encoding.UTF8.GetByteCount(convstr); | ||||
|             var buffer = Marshal.AllocHGlobal(convbytecnt); | ||||
|             var convbytecnt = Encoding.UTF8.GetByteCount(input); | ||||
|             var buffer = Marshal.AllocHGlobal(convbytecnt + 1); | ||||
|             for (int i = 0; i < convbytecnt + 1; i++) | ||||
|             { | ||||
|                 Marshal.WriteByte(buffer, i, 0); | ||||
|             } | ||||
|             _buffers.Add(buffer); | ||||
|             Marshal.Copy(Encoding.UTF8.GetBytes(convstr), 0, buffer, convbytecnt); | ||||
|             Marshal.Copy(Encoding.UTF8.GetBytes(input), 0, buffer, convbytecnt); | ||||
|             return buffer; | ||||
|         } | ||||
|  | ||||
| @@ -181,30 +216,6 @@ public class DiscordRpc | ||||
|             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> | ||||
|         /// Free the allocated memory for conversion to <see cref="RichPresenceStruct"/> | ||||
|         /// </summary> | ||||
|   | ||||
| @@ -46,8 +46,8 @@ public class ScriptBatch | ||||
| 		string[] srcDlls = { "../../builds/install/osx-dynamic/lib/libdiscord-rpc.dylib" }; | ||||
| 		#else | ||||
| 		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[] srcDlls = { "../../builds/install/linux-dynamic/bin/discord-rpc.dll", "../../builds/install/win64-dynamic/bin/discord-rpc.dll" }; | ||||
| 		string[] dstDlls = { "Assets/Plugins/discord-rpc.so" }; | ||||
| 		string[] srcDlls = { "../../builds/install/linux-dynamic/lib/libdiscord-rpc.so" }; | ||||
| 		#endif | ||||
|  | ||||
| 		Debug.Assert(dstDlls.Length == srcDlls.Length); | ||||
|   | ||||
| @@ -47,19 +47,24 @@ static void updateDiscordPresence() | ||||
|         discordPresence.partyId = "party1234"; | ||||
|         discordPresence.partySize = 1; | ||||
|         discordPresence.partyMax = 6; | ||||
|         discordPresence.partyPrivacy = DISCORD_PARTY_PUBLIC; | ||||
|         discordPresence.matchSecret = "xyzzy"; | ||||
|         discordPresence.joinSecret = "join"; | ||||
|         discordPresence.spectateSecret = "look"; | ||||
|         discordPresence.instance = 0; | ||||
|         Discord_UpdatePresence(&discordPresence); | ||||
|     } else { | ||||
|     } | ||||
|     else { | ||||
|         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) | ||||
| @@ -82,13 +87,13 @@ static void handleDiscordSpectate(const char* secret) | ||||
|     printf("\nDiscord: spectate (%s)\n", secret); | ||||
| } | ||||
|  | ||||
| static void handleDiscordJoinRequest(const DiscordJoinRequest* request) | ||||
| static void handleDiscordJoinRequest(const DiscordUser* request) | ||||
| { | ||||
|     int response = -1; | ||||
|     char yn[4]; | ||||
|     printf("\nDiscord: join request from %s - %s - %s\n", | ||||
|     printf("\nDiscord: join request from %s#%s - %s\n", | ||||
|            request->username, | ||||
|            request->avatar, | ||||
|            request->discriminator, | ||||
|            request->userId); | ||||
|     do { | ||||
|         printf("Accept? (y/n)"); | ||||
| @@ -152,7 +157,8 @@ static void gameLoop() | ||||
|                 if (SendPresence) { | ||||
|                     printf("Clearing presence information.\n"); | ||||
|                     SendPresence = 0; | ||||
|                 } else { | ||||
|                 } | ||||
|                 else { | ||||
|                     printf("Restoring presence information.\n"); | ||||
|                     SendPresence = 1; | ||||
|                 } | ||||
|   | ||||
										
											Binary file not shown.
										
									
								
							| @@ -12,19 +12,26 @@ void FDiscordRpcModule::StartupModule() | ||||
| #if defined(DISCORD_DYNAMIC_LIB) | ||||
|     // Get the base directory of this plugin | ||||
|     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 | ||||
|     const FString LibName = TEXT("discord-rpc"); | ||||
|     const FString LibDir = FPaths::Combine(*SDKDir, TEXT("Win64")); | ||||
|     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); | ||||
|     } | ||||
| #elif PLATFORM_MAC | ||||
|     const FString LibName = TEXT("libdiscord-rpc"); | ||||
|     const FString LibDir = FPaths::Combine(*SDKDir, TEXT("Mac")); | ||||
|     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); | ||||
|     } | ||||
| #endif | ||||
| @@ -49,8 +56,7 @@ bool FDiscordRpcModule::LoadDependency(const FString& Dir, const FString& Name, | ||||
|  | ||||
|     Handle = FPlatformProcess::GetDllHandle(*Path); | ||||
|  | ||||
|     if (Handle == nullptr) | ||||
|     { | ||||
|     if (Handle == nullptr) { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
| @@ -59,8 +65,7 @@ bool FDiscordRpcModule::LoadDependency(const FString& Dir, const FString& Name, | ||||
|  | ||||
| void FDiscordRpcModule::FreeDependency(void*& Handle) | ||||
| { | ||||
|     if (Handle != nullptr) | ||||
|     { | ||||
|     if (Handle != nullptr) { | ||||
|         FPlatformProcess::FreeDllHandle(Handle); | ||||
|         Handle = nullptr; | ||||
|     } | ||||
|   | ||||
| @@ -6,12 +6,22 @@ DEFINE_LOG_CATEGORY(Discord) | ||||
|  | ||||
| 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) { | ||||
|         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; | ||||
|     jr.userId = ANSI_TO_TCHAR(request->userId); | ||||
|     jr.username = ANSI_TO_TCHAR(request->username); | ||||
|     jr.discriminator = ANSI_TO_TCHAR(request->discriminator); | ||||
|     jr.avatar = ANSI_TO_TCHAR(request->avatar); | ||||
|     UE_LOG(Discord, Log, TEXT("Discord join request from %s - %s#%s"), *jr.userId, *jr.username, *jr.discriminator); | ||||
|     FDiscordUserData ud; | ||||
|     ud.userId = ANSI_TO_TCHAR(request->userId); | ||||
|     ud.username = ANSI_TO_TCHAR(request->username); | ||||
|     ud.discriminator = ANSI_TO_TCHAR(request->discriminator); | ||||
|     ud.avatar = ANSI_TO_TCHAR(request->avatar); | ||||
|     UE_LOG(Discord, | ||||
|            Log, | ||||
|            TEXT("Discord join request from %s - %s#%s"), | ||||
|            *ud.userId, | ||||
|            *ud.username, | ||||
|            *ud.discriminator); | ||||
|     if (self) { | ||||
|         self->OnJoinRequest.Broadcast(jr); | ||||
|         self->OnJoinRequest.Broadcast(ud); | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -138,6 +153,7 @@ void UDiscordRpc::UpdatePresence() | ||||
|     rp.endTimestamp = RichPresence.endTimestamp; | ||||
|     rp.partySize = RichPresence.partySize; | ||||
|     rp.partyMax = RichPresence.partyMax; | ||||
|     rp.partyPrivacy = (int)RichPresence.partyPrivacy; | ||||
|     rp.instance = RichPresence.instance; | ||||
|  | ||||
|     Discord_UpdatePresence(&rp); | ||||
|   | ||||
| @@ -11,7 +11,7 @@ | ||||
| * Ask to join callback data | ||||
| */ | ||||
| USTRUCT(BlueprintType) | ||||
| struct FDiscordJoinRequestData { | ||||
| struct FDiscordUserData { | ||||
|     GENERATED_USTRUCT_BODY() | ||||
|  | ||||
|     UPROPERTY(BlueprintReadOnly) | ||||
| @@ -24,15 +24,35 @@ struct FDiscordJoinRequestData { | ||||
|     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_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(FDiscordErrored, int, errorCode, const FString&, errorMessage); | ||||
| DECLARE_DYNAMIC_MULTICAST_DELEGATE_OneParam(FDiscordJoin, const FString&, joinSecret); | ||||
| 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 | ||||
|  | ||||
| @@ -67,6 +87,8 @@ struct FDiscordRichPresence { | ||||
|     UPROPERTY(BlueprintReadWrite) | ||||
|     int partyMax; | ||||
|     UPROPERTY(BlueprintReadWrite) | ||||
|     EDiscordPartyPrivacy partyPrivacy; | ||||
|     UPROPERTY(BlueprintReadWrite) | ||||
|     FString matchSecret; | ||||
|     UPROPERTY(BlueprintReadWrite) | ||||
|     FString joinSecret; | ||||
|   | ||||
| @@ -1,17 +1,17 @@ | ||||
| #pragma once | ||||
|  | ||||
| #if defined(DISCORD_DYNAMIC_LIB) | ||||
| #  if defined(_WIN32) | ||||
| #    if defined(DISCORD_BUILDING_SDK) | ||||
| #      define DISCORD_EXPORT __declspec(dllexport) | ||||
| #    else | ||||
| #      define DISCORD_EXPORT __declspec(dllimport) | ||||
| #    endif | ||||
| #  else | ||||
| #    define DISCORD_EXPORT __attribute__((visibility("default"))) | ||||
| #  endif | ||||
| #if defined(_WIN32) | ||||
| #if defined(DISCORD_BUILDING_SDK) | ||||
| #define DISCORD_EXPORT __declspec(dllexport) | ||||
| #else | ||||
| #  define DISCORD_EXPORT | ||||
| #define DISCORD_EXPORT __declspec(dllimport) | ||||
| #endif | ||||
| #else | ||||
| #define DISCORD_EXPORT __attribute__((visibility("default"))) | ||||
| #endif | ||||
| #else | ||||
| #define DISCORD_EXPORT | ||||
| #endif | ||||
|  | ||||
| #ifdef __cplusplus | ||||
|   | ||||
| @@ -35,31 +35,34 @@ typedef struct DiscordRichPresence { | ||||
|     const char* partyId;        /* max 128 bytes */ | ||||
|     int partySize; | ||||
|     int partyMax; | ||||
|     int partyPrivacy; | ||||
|     const char* matchSecret;    /* max 128 bytes */ | ||||
|     const char* joinSecret;     /* max 128 bytes */ | ||||
|     const char* spectateSecret; /* max 128 bytes */ | ||||
|     int8_t instance; | ||||
| } DiscordRichPresence; | ||||
|  | ||||
| typedef struct DiscordJoinRequest { | ||||
| typedef struct DiscordUser { | ||||
|     const char* userId; | ||||
|     const char* username; | ||||
|     const char* discriminator; | ||||
|     const char* avatar; | ||||
| } DiscordJoinRequest; | ||||
| } DiscordUser; | ||||
|  | ||||
| typedef struct DiscordEventHandlers { | ||||
|     void (*ready)(void); | ||||
|     void (*ready)(const DiscordUser* request); | ||||
|     void (*disconnected)(int errorCode, const char* message); | ||||
|     void (*errored)(int errorCode, const char* message); | ||||
|     void (*joinGame)(const char* joinSecret); | ||||
|     void (*spectateGame)(const char* spectateSecret); | ||||
|     void (*joinRequest)(const DiscordJoinRequest* request); | ||||
|     void (*joinRequest)(const DiscordUser* request); | ||||
| } DiscordEventHandlers; | ||||
|  | ||||
| #define DISCORD_REPLY_NO 0 | ||||
| #define DISCORD_REPLY_YES 1 | ||||
| #define DISCORD_REPLY_IGNORE 2 | ||||
| #define DISCORD_PARTY_PRIVATE 0 | ||||
| #define DISCORD_PARTY_PUBLIC 1 | ||||
|  | ||||
| DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | ||||
|                                        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(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) | ||||
|  | ||||
| @@ -71,12 +72,23 @@ if(UNIX) | ||||
|  | ||||
|     add_library(discord-rpc ${BASE_RPC_SRC}) | ||||
|     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 | ||||
|         -g | ||||
|         -Wall | ||||
|         -Wextra | ||||
|         -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-old-style-cast # it's fine | ||||
|         -Wno-c++98-compat # that was almost 2 decades ago | ||||
| @@ -133,3 +145,12 @@ install( | ||||
| 		"../include/discord_register.h" | ||||
|     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(); | ||||
|     } | ||||
|     else if (res == 0) { | ||||
|         Close(); | ||||
|     } | ||||
|     return res == (int)length; | ||||
| } | ||||
|   | ||||
| @@ -33,13 +33,15 @@ extern "C" DISCORD_EXPORT void Discord_Register(const char* applicationId, const | ||||
|  | ||||
|     char exePath[1024]; | ||||
|     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; | ||||
|         } | ||||
|         exePath[size] = '\0'; | ||||
|         command = exePath; | ||||
|     } | ||||
|  | ||||
|     const char* destopFileFormat = "[Desktop Entry]\n" | ||||
|     const char* desktopFileFormat = "[Desktop Entry]\n" | ||||
|                                    "Name=Game %s\n" | ||||
|                                    "Exec=%s %%u\n" // note: it really wants that %u in there | ||||
|                                    "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"; | ||||
|     char desktopFile[2048]; | ||||
|     int fileLen = snprintf( | ||||
|       desktopFile, sizeof(desktopFile), destopFileFormat, applicationId, command, applicationId); | ||||
|       desktopFile, sizeof(desktopFile), desktopFileFormat, applicationId, command, applicationId); | ||||
|     if (fileLen <= 0) { | ||||
|         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]; | ||||
|     sprintf(command, "xdg-open steam://rungameid/%s", steamId); | ||||
|   | ||||
| @@ -7,7 +7,6 @@ | ||||
| #define NOIME | ||||
| #include <windows.h> | ||||
| #include <psapi.h> | ||||
| #include <cwchar> | ||||
| #include <cstdio> | ||||
|  | ||||
| /** | ||||
| @@ -20,19 +19,22 @@ | ||||
|  * The entire function is rewritten | ||||
|  */ | ||||
| #ifdef __MINGW32__ | ||||
| #include <wchar.h> | ||||
| /// strsafe.h fixes | ||||
| static HRESULT StringCbPrintfW(LPWSTR pszDest, size_t cbDest, LPCWSTR pszFormat, ...) | ||||
| { | ||||
|     HRESULT ret; | ||||
|     va_list va; | ||||
|     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); | ||||
|     pszDest[cbDest - 1] = 0; // Terminate the string in case a buffer overflow; -1 will be returned | ||||
|     va_end(va); | ||||
|     return ret; | ||||
| } | ||||
| #else | ||||
| #include <cwchar> | ||||
| #include <strsafe.h> | ||||
| #endif // __MINGW32__ | ||||
|  | ||||
| @@ -44,17 +46,24 @@ static HRESULT StringCbPrintfW(LPWSTR pszDest, size_t cbDest, LPCWSTR pszFormat, | ||||
| #undefine RegSetKeyValueW | ||||
| #endif | ||||
| #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; | ||||
|     LSTATUS ret; | ||||
|     if (subkey && subkey[0]) | ||||
|     { | ||||
|         if((ret = RegCreateKeyExW(hkey, subkey, 0, 0, 0, KEY_ALL_ACCESS, 0, &hsubkey, 0)) != ERROR_SUCCESS) return ret; | ||||
|     if (subkey && subkey[0]) { | ||||
|         if ((ret = RegCreateKeyExW(hkey, subkey, 0, 0, 0, KEY_ALL_ACCESS, 0, &hsubkey, 0)) != | ||||
|             ERROR_SUCCESS) | ||||
|             return ret; | ||||
|         htkey = hsubkey; | ||||
|     } | ||||
|     ret = RegSetValueExW(htkey, name, 0, type, (const BYTE*)data, len); | ||||
|     if (hsubkey && hsubkey != hkey) RegCloseKey(hsubkey); | ||||
|     if (hsubkey && hsubkey != hkey) | ||||
|         RegCloseKey(hsubkey); | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| @@ -72,7 +81,7 @@ static void Discord_RegisterW(const wchar_t* applicationId, const wchar_t* comma | ||||
|         StringCbPrintfW(openCommand, sizeof(openCommand), L"%s", command); | ||||
|     } | ||||
|     else { | ||||
|         //StringCbCopyW(openCommand, sizeof(openCommand), exeFilePath); | ||||
|         // StringCbCopyW(openCommand, sizeof(openCommand), exeFilePath); | ||||
|         StringCbPrintfW(openCommand, sizeof(openCommand), L"%s", exeFilePath); | ||||
|     } | ||||
|  | ||||
| @@ -138,7 +147,8 @@ extern "C" DISCORD_EXPORT void Discord_Register(const char* applicationId, const | ||||
|     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]; | ||||
|     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 | ||||
|     // terminator = 21 | ||||
|     char userId[32]; | ||||
| @@ -47,12 +47,14 @@ struct JoinRequest { | ||||
| }; | ||||
|  | ||||
| static RpcConnection* Connection{nullptr}; | ||||
| static DiscordEventHandlers QueuedHandlers{}; | ||||
| static DiscordEventHandlers Handlers{}; | ||||
| static std::atomic_bool WasJustConnected{false}; | ||||
| static std::atomic_bool WasJustDisconnected{false}; | ||||
| static std::atomic_bool GotErrorMessage{false}; | ||||
| static std::atomic_bool WasJoinGame{false}; | ||||
| static std::atomic_bool WasSpectateGame{false}; | ||||
| static std::atomic_bool UpdatePresence{false}; | ||||
| static char JoinGameSecret[256]; | ||||
| static char SpectateGameSecret[256]; | ||||
| static int LastErrorCode{0}; | ||||
| @@ -63,7 +65,8 @@ static std::mutex PresenceMutex; | ||||
| static std::mutex HandlerMutex; | ||||
| static QueuedMessage QueuedPresence{}; | ||||
| 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 | ||||
| // backoff from 0.5 seconds to 1 minute | ||||
| @@ -87,10 +90,11 @@ public: | ||||
|         keepRunning.store(true); | ||||
|         ioThread = std::thread([&]() { | ||||
|             const std::chrono::duration<int64_t, std::milli> maxWait{500LL}; | ||||
|             while (keepRunning.load()) { | ||||
|             Discord_UpdateConnection(); | ||||
|             while (keepRunning.load()) { | ||||
|                 std::unique_lock<std::mutex> lock(waitForIOMutex); | ||||
|                 waitForIOActivity.wait_for(lock, maxWait); | ||||
|                 Discord_UpdateConnection(); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
| @@ -116,7 +120,7 @@ public: | ||||
|     void Notify() {} | ||||
| }; | ||||
| #endif // DISCORD_DISABLE_IO_THREAD | ||||
| static IoThreadHolder IoThread; | ||||
| static IoThreadHolder* IoThread{nullptr}; | ||||
|  | ||||
| static void UpdateReconnectTime() | ||||
| { | ||||
| @@ -211,17 +215,17 @@ static void Discord_UpdateConnection(void) | ||||
|         } | ||||
|  | ||||
|         // writes | ||||
|         if (QueuedPresence.length) { | ||||
|         if (UpdatePresence.exchange(false) && QueuedPresence.length) { | ||||
|             QueuedMessage local; | ||||
|             { | ||||
|                 std::lock_guard<std::mutex> guard(PresenceMutex); | ||||
|                 local.Copy(QueuedPresence); | ||||
|                 QueuedPresence.length = 0; | ||||
|             } | ||||
|             if (!Connection->Write(local.buffer, local.length)) { | ||||
|                 // if we fail to send, requeue | ||||
|                 std::lock_guard<std::mutex> guard(PresenceMutex); | ||||
|                 QueuedPresence.Copy(local); | ||||
|                 UpdatePresence.exchange(true); | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -235,7 +239,9 @@ static void Discord_UpdateConnection(void) | ||||
|  | ||||
| static void SignalIOActivity() | ||||
| { | ||||
|     IoThread.Notify(); | ||||
|     if (IoThread != nullptr) { | ||||
|         IoThread->Notify(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static bool RegisterForEvent(const char* evtName) | ||||
| @@ -269,6 +275,11 @@ extern "C" DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | ||||
|                                                   int autoRegister, | ||||
|                                                   const char* optionalSteamId) | ||||
| { | ||||
|     IoThread = new (std::nothrow) IoThreadHolder(); | ||||
|     if (IoThread == nullptr) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (autoRegister) { | ||||
|         if (optionalSteamId && optionalSteamId[0]) { | ||||
|             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); | ||||
|  | ||||
|         if (handlers) { | ||||
|             Handlers = *handlers; | ||||
|             QueuedHandlers = *handlers; | ||||
|         } | ||||
|         else { | ||||
|             Handlers = {}; | ||||
|             QueuedHandlers = {}; | ||||
|         } | ||||
|  | ||||
|         Handlers = {}; | ||||
|     } | ||||
|  | ||||
|     if (Connection) { | ||||
| @@ -295,8 +309,31 @@ extern "C" DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | ||||
|     } | ||||
|  | ||||
|     Connection = RpcConnection::Create(applicationId); | ||||
|     Connection->onConnect = []() { | ||||
|         Discord_UpdateHandlers(&Handlers); | ||||
|     Connection->onConnect = [](JsonDocument& readyMessage) { | ||||
|         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); | ||||
|         ReconnectTimeMs.reset(); | ||||
|     }; | ||||
| @@ -307,7 +344,7 @@ extern "C" DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | ||||
|         UpdateReconnectTime(); | ||||
|     }; | ||||
|  | ||||
|     IoThread.Start(); | ||||
|     IoThread->Start(); | ||||
| } | ||||
|  | ||||
| extern "C" DISCORD_EXPORT void Discord_Shutdown(void) | ||||
| @@ -318,7 +355,14 @@ extern "C" DISCORD_EXPORT void Discord_Shutdown(void) | ||||
|     Connection->onConnect = nullptr; | ||||
|     Connection->onDisconnect = nullptr; | ||||
|     Handlers = {}; | ||||
|     IoThread.Stop(); | ||||
|     QueuedPresence.length = 0; | ||||
|     UpdatePresence.exchange(false); | ||||
|     if (IoThread != nullptr) { | ||||
|         IoThread->Stop(); | ||||
|         delete IoThread; | ||||
|         IoThread = nullptr; | ||||
|     } | ||||
|  | ||||
|     RpcConnection::Destroy(Connection); | ||||
| } | ||||
|  | ||||
| @@ -328,6 +372,7 @@ extern "C" DISCORD_EXPORT void Discord_UpdatePresence(const DiscordRichPresence* | ||||
|         std::lock_guard<std::mutex> guard(PresenceMutex); | ||||
|         QueuedPresence.length = JsonWriteRichPresenceObj( | ||||
|           QueuedPresence.buffer, sizeof(QueuedPresence.buffer), Nonce++, Pid, presence); | ||||
|         UpdatePresence.exchange(true); | ||||
|     } | ||||
|     SignalIOActivity(); | ||||
| } | ||||
| @@ -376,7 +421,11 @@ extern "C" DISCORD_EXPORT void Discord_RunCallbacks(void) | ||||
|     if (WasJustConnected.exchange(false)) { | ||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||
|         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); | ||||
|             if (Handlers.joinRequest) { | ||||
|                 DiscordJoinRequest djr{req->userId, req->username, req->discriminator, req->avatar}; | ||||
|                 Handlers.joinRequest(&djr); | ||||
|                 DiscordUser du{req->userId, req->username, req->discriminator, req->avatar}; | ||||
|                 Handlers.joinRequest(&du); | ||||
|             } | ||||
|         } | ||||
|         JoinAskQueue.CommitSend(); | ||||
| @@ -430,7 +479,6 @@ extern "C" DISCORD_EXPORT void Discord_RunCallbacks(void) | ||||
| extern "C" DISCORD_EXPORT void Discord_UpdateHandlers(DiscordEventHandlers* newHandlers) | ||||
| { | ||||
|     if (newHandlers) { | ||||
|  | ||||
| #define HANDLE_EVENT_REGISTRATION(handler_name, event)              \ | ||||
|     if (!Handlers.handler_name && newHandlers->handler_name) {      \ | ||||
|         RegisterForEvent(event);                                    \ | ||||
| @@ -448,8 +496,7 @@ extern "C" DISCORD_EXPORT void Discord_UpdateHandlers(DiscordEventHandlers* newH | ||||
|  | ||||
|         Handlers = *newHandlers; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|     else { | ||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||
|         Handlers = {}; | ||||
|     } | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
|  | ||||
| template <typename ElementType, size_t QueueSize> | ||||
| class MsgQueue { | ||||
|     ElementType queue_[QueueSize]{}; | ||||
|     ElementType queue_[QueueSize]; | ||||
|     std::atomic_uint nextAdd_{0}; | ||||
|     std::atomic_uint nextSend_{0}; | ||||
|     std::atomic_uint pendingSends_{0}; | ||||
|   | ||||
| @@ -26,13 +26,9 @@ void RpcConnection::Open() | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (state == State::Disconnected) { | ||||
|         if (connection->Open()) { | ||||
|         } | ||||
|         else { | ||||
|     if (state == State::Disconnected && !connection->Open()) { | ||||
|         return; | ||||
|     } | ||||
|     } | ||||
|  | ||||
|     if (state == State::SentHandshake) { | ||||
|         JsonDocument message; | ||||
| @@ -42,7 +38,7 @@ void RpcConnection::Open() | ||||
|             if (cmd && evt && !strcmp(cmd, "DISPATCH") && !strcmp(evt, "READY")) { | ||||
|                 state = State::Connected; | ||||
|                 if (onConnect) { | ||||
|                     onConnect(); | ||||
|                     onConnect(message); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|   | ||||
| @@ -40,7 +40,7 @@ struct RpcConnection { | ||||
|  | ||||
|     BaseConnection* connection{nullptr}; | ||||
|     State state{State::Disconnected}; | ||||
|     void (*onConnect)(){nullptr}; | ||||
|     void (*onConnect)(JsonDocument& message){nullptr}; | ||||
|     void (*onDisconnect)(int errorCode, const char* message){nullptr}; | ||||
|     char appId[64]{}; | ||||
|     int lastErrorCode{0}; | ||||
|   | ||||
| @@ -102,8 +102,7 @@ size_t JsonWriteRichPresenceObj(char* dest, | ||||
|             WriteKey(writer, "pid"); | ||||
|             writer.Int(pid); | ||||
|  | ||||
|             if (presence != nullptr) | ||||
|             { | ||||
|             if (presence != nullptr) { | ||||
|                 WriteObject activity(writer, "activity"); | ||||
|  | ||||
|                 WriteOptionalString(writer, "state", presence->state); | ||||
| @@ -135,7 +134,7 @@ size_t JsonWriteRichPresenceObj(char* dest, | ||||
|                 } | ||||
|  | ||||
|                 if ((presence->partyId && presence->partyId[0]) || presence->partySize || | ||||
|                     presence->partyMax) { | ||||
|                     presence->partyMax || presence->partyPrivacy) { | ||||
|                     WriteObject party(writer, "party"); | ||||
|                     WriteOptionalString(writer, "id", presence->partyId); | ||||
|                     if (presence->partySize && presence->partyMax) { | ||||
| @@ -143,6 +142,11 @@ size_t JsonWriteRichPresenceObj(char* dest, | ||||
|                         writer.Int(presence->partySize); | ||||
|                         writer.Int(presence->partyMax); | ||||
|                     } | ||||
|  | ||||
|                     if (presence->partyPrivacy) { | ||||
|                         WriteKey(writer, "privacy"); | ||||
|                         writer.Int(presence->partyPrivacy); | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 if ((presence->matchSecret && presence->matchSecret[0]) || | ||||
|   | ||||
		Reference in New Issue
	
	Block a user