Compare commits
	
		
			50 Commits
		
	
	
		
			v3.0.0
			...
			choose-pip
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 8bb85f0545 | ||
|  | 766596722c | ||
|  | 7fe88765fd | ||
|  | 1d30b94987 | ||
|  | 6796d2ffa9 | ||
|  | d90a8efd47 | ||
|  | 544f91a5a8 | ||
|  | 2f52c24f6d | ||
|  | d5a342c7bb | ||
|  | 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 | ||
|  | 087282cd4b | ||
|  | 7e5d57e6fd | ||
|  | f3bd411b99 | ||
|  | 8e0c7848a6 | ||
|  | e7f9396807 | ||
|  | ad0b844672 | ||
|  | d279c24c6a | ||
|  | d9caf72e9a | 
| @@ -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 | ||||
|   | ||||
| @@ -28,23 +28,23 @@ endif(CLANG_FORMAT_CMD) | ||||
|  | ||||
| # thirdparty stuff | ||||
| execute_process( | ||||
|     COMMAND mkdir ${CMAKE_SOURCE_DIR}/thirdparty | ||||
|     COMMAND mkdir ${CMAKE_CURRENT_SOURCE_DIR}/thirdparty | ||||
|     ERROR_QUIET | ||||
| ) | ||||
|  | ||||
| find_file(RAPIDJSONTEST NAMES rapidjson rapidjson-1.1.0 PATHS ${CMAKE_SOURCE_DIR}/thirdparty CMAKE_FIND_ROOT_PATH_BOTH) | ||||
| find_file(RAPIDJSONTEST NAMES rapidjson rapidjson-1.1.0 PATHS ${CMAKE_CURRENT_SOURCE_DIR}/thirdparty CMAKE_FIND_ROOT_PATH_BOTH) | ||||
| if (NOT RAPIDJSONTEST) | ||||
|     message("no rapidjson, download") | ||||
|     set(RJ_TAR_FILE ${CMAKE_SOURCE_DIR}/thirdparty/v1.1.0.tar.gz) | ||||
|     set(RJ_TAR_FILE ${CMAKE_CURRENT_SOURCE_DIR}/thirdparty/v1.1.0.tar.gz) | ||||
|     file(DOWNLOAD https://github.com/miloyip/rapidjson/archive/v1.1.0.tar.gz ${RJ_TAR_FILE}) | ||||
|     execute_process( | ||||
|         COMMAND ${CMAKE_COMMAND} -E tar xzf ${RJ_TAR_FILE} | ||||
|         WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}/thirdparty | ||||
|         WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/thirdparty | ||||
|     ) | ||||
|     file(REMOVE ${RJ_TAR_FILE}) | ||||
| endif(NOT RAPIDJSONTEST) | ||||
|  | ||||
| find_file(RAPIDJSON NAMES rapidjson rapidjson-1.1.0 PATHS ${CMAKE_SOURCE_DIR}/thirdparty CMAKE_FIND_ROOT_PATH_BOTH) | ||||
| find_file(RAPIDJSON NAMES rapidjson rapidjson-1.1.0 PATHS ${CMAKE_CURRENT_SOURCE_DIR}/thirdparty CMAKE_FIND_ROOT_PATH_BOTH) | ||||
|  | ||||
| add_library(rapidjson STATIC IMPORTED ${RAPIDJSON}) | ||||
|  | ||||
|   | ||||
							
								
								
									
										91
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										91
									
								
								README.md
									
									
									
									
									
								
							| @@ -7,7 +7,7 @@ have callbacks for where a more complete game would do more things (joining, spe | ||||
|  | ||||
| ## Documentation | ||||
|  | ||||
| The most up to date documentation for Rich Presence can always be found in our [developer site](https://discordapp.com/developers/docs/rich-presence/how-to)! | ||||
| The most up to date documentation for Rich Presence can always be found on our [developer site](https://discordapp.com/developers/docs/rich-presence/how-to)! If you're interested in rolling your own native implementation of Rich Presence via IPC sockets instead of using our SDK—hey, you've got free time, right?—check out the ["Hard Mode" documentation](https://github.com/discordapp/discord-rpc/blob/master/documentation/hard-mode.md). | ||||
|  | ||||
| ## Basic Usage | ||||
|  | ||||
| @@ -15,13 +15,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 +92,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 +100,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 +139,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)        | | ||||
|   | ||||
							
								
								
									
										49
									
								
								build.py
									
									
									
									
									
								
							
							
						
						
									
										49
									
								
								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') | ||||
| @@ -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') | ||||
| @@ -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: | ||||
| @@ -224,22 +207,28 @@ def sign(): | ||||
|         sign_command_base = [ | ||||
|             tool, | ||||
|             'sign', | ||||
|             '/n', 'Hammer & Chisel 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' | ||||
|   | ||||
| @@ -93,8 +93,7 @@ And third is the `ACTIVITY_JOIN_REQUEST` event: | ||||
|       "username": "Mason", | ||||
|       "discriminator": "1337", | ||||
|       "avatar": "a_bab14f271d565501444b2ca3be944b25" | ||||
|     }, | ||||
|     "secret": "e459ca99273f59909dd16ed97865f3ad" | ||||
|     } | ||||
|   }, | ||||
|   "evt": "ACTIVITY_JOIN_REQUEST" | ||||
| } | ||||
| @@ -125,3 +124,41 @@ In order to receive these events, you need to [subscribe](https://discordapp.com | ||||
|     "cmd": "SUBSCRIBE" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| To unsubscribe from these events, resend with the command `UNSUBSCRIBE` | ||||
|  | ||||
| ## Responding | ||||
| A discord user will request access to the game. If the ACTIVITY_JOIN_REQUEST has been subscribed too, the ACTIVITY_JOIN_REQUEST event will be sent to the host's game. Accept it with following model: | ||||
| ```json | ||||
| { | ||||
|     "nonce": "5dc0c062-98c6-47a0-8922-15aerg126", | ||||
|     "cmd": "SEND_ACTIVITY_JOIN_INVITE", | ||||
|     "args":  | ||||
|     { | ||||
|         "user_id": "53908232506183680" | ||||
|     } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| To reject the request, use `CLOSE_ACTIVITY_REQUEST`: | ||||
| ```json | ||||
| { | ||||
|     "nonce": "5dc0c062-98c6-47a0-8922-dasg256eafg", | ||||
|     "cmd": "CLOSE_ACTIVITY_REQUEST", | ||||
|     "args":  | ||||
|     { | ||||
|         "user_id": "53908232506183680" | ||||
|     } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## Notes | ||||
| Here are just some quick notes to help with some common troubleshooting problems. | ||||
| * IPC will echo back every command you send as a response. Use this as a lock-step feature to avoid flooding messages. Can be used to validate messages such as the Presence or Subscribes. | ||||
| * The pipe expects for frames to be written in a single byte array. You cannot do multiple `stream.Write(opcode);` `stream.Write(length);` as it will break the pipe. Instead create a buffer, write the data to the buffer, then send the entire buffer to the stream. | ||||
| * Discord can be on any pipe ranging from `discord-ipc-0` to `discord-ipc-9`. It is a good idea to try and connect to each one and keeping the first one you connect too. For multiple clients (eg Discord and Canary), you might want to add a feature to manually select the pipe so you can more easily debug the application. | ||||
| * All enums are `lower_snake_case`.  | ||||
| * The opcode and length in the header are `Little Endian Unsigned Integers (32bits)`. In some languages, you must convert them as they can be architecture specific. | ||||
| * [Discord Rich Presence How-To](https://discordapp.com/developers/docs/rich-presence/how-to) contains a lot of the information this document doesn't. For example, it will tell you about the response payload. | ||||
| * In the documentation, DISCORD_REPLY_IGNORE is just implemented the same as DISCORD_REPLY_NO. | ||||
| * You can test the Join / Spectate feature by enabling them in your profile and whitelisting a test account. Use Canary to run 2 accounts on the same machine. | ||||
|   | ||||
| @@ -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; | ||||
| @@ -50,43 +49,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 +97,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)] | ||||
| @@ -54,7 +63,7 @@ public class DiscordRpc | ||||
|     } | ||||
|  | ||||
|     [Serializable] | ||||
|     public struct JoinRequest | ||||
|     public struct DiscordUser | ||||
|     { | ||||
|         public string userId; | ||||
|         public string username; | ||||
| @@ -69,8 +78,23 @@ public class DiscordRpc | ||||
|         Ignore = 2 | ||||
|     } | ||||
|  | ||||
|     public static void Initialize(string applicationId, ref EventHandlers handlers, bool autoRegister, string optionalSteamId, int pipe = 0) | ||||
|     { | ||||
|         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, pipe); | ||||
|     } | ||||
|  | ||||
|     [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, int pipe); | ||||
|  | ||||
|     [DllImport("discord-rpc", EntryPoint = "Discord_Shutdown", CallingConvention = CallingConvention.Cdecl)] | ||||
|     public static extern void Shutdown(); | ||||
| @@ -87,6 +111,9 @@ public class DiscordRpc | ||||
|     [DllImport("discord-rpc", EntryPoint = "Discord_Respond", CallingConvention = CallingConvention.Cdecl)] | ||||
|     public static extern void Respond(string userId, Reply reply); | ||||
|  | ||||
|     [DllImport("discord-rpc", EntryPoint = "Discord_UpdateHandlers", CallingConvention = CallingConvention.Cdecl)] | ||||
|     public static extern void UpdateHandlers(ref EventHandlers handlers); | ||||
|  | ||||
|     public static void UpdatePresence(RichPresence presence) | ||||
|     { | ||||
|         var presencestruct = presence.GetStruct(); | ||||
| @@ -126,20 +153,20 @@ 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.matchSecret = StrToPtr(matchSecret); | ||||
|             _presence.joinSecret = StrToPtr(joinSecret); | ||||
|             _presence.spectateSecret = StrToPtr(spectateSecret); | ||||
|             _presence.instance = instance; | ||||
|  | ||||
|             return _presence; | ||||
| @@ -149,16 +176,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; | ||||
|         } | ||||
|  | ||||
| @@ -178,30 +207,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); | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| /* | ||||
|     This is a simple example in C of using the rich presence API asyncronously. | ||||
|     This is a simple example in C of using the rich presence API asynchronously. | ||||
| */ | ||||
|  | ||||
| #define _CRT_SECURE_NO_WARNINGS /* thanks Microsoft */ | ||||
| @@ -52,14 +52,18 @@ static void updateDiscordPresence() | ||||
|         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 +86,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)"); | ||||
| @@ -125,7 +129,7 @@ static void discordInit() | ||||
|     handlers.joinGame = handleDiscordJoin; | ||||
|     handlers.spectateGame = handleDiscordSpectate; | ||||
|     handlers.joinRequest = handleDiscordJoinRequest; | ||||
|     Discord_Initialize(APPLICATION_ID, &handlers, 1, NULL); | ||||
|     Discord_Initialize(APPLICATION_ID, &handlers, 1, NULL, 0); | ||||
| } | ||||
|  | ||||
| static void gameLoop() | ||||
| @@ -152,7 +156,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,22 +62,28 @@ 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); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void UDiscordRpc::Initialize(const FString& applicationId, | ||||
|                              bool autoRegister, | ||||
|     const FString& optionalSteamId) | ||||
|                              const FString& optionalSteamId, | ||||
|                              int pipe) | ||||
| { | ||||
|     self = this; | ||||
|     IsConnected = false; | ||||
| @@ -87,7 +103,7 @@ void UDiscordRpc::Initialize(const FString& applicationId, | ||||
|     auto appId = StringCast<ANSICHAR>(*applicationId); | ||||
|     auto steamId = StringCast<ANSICHAR>(*optionalSteamId); | ||||
|     Discord_Initialize( | ||||
|         (const char*)appId.Get(), &handlers, autoRegister, (const char*)steamId.Get()); | ||||
|       (const char*)appId.Get(), &handlers, autoRegister, (const char*)steamId.Get(), pipe); | ||||
| } | ||||
|  | ||||
| void UDiscordRpc::Shutdown() | ||||
| @@ -134,6 +150,8 @@ void UDiscordRpc::UpdatePresence() | ||||
|  | ||||
|     auto spectateSecret = StringCast<ANSICHAR>(*RichPresence.spectateSecret); | ||||
|     rp.spectateSecret = spectateSecret.Get(); | ||||
|     rp.startTimestamp = RichPresence.startTimestamp; | ||||
|     rp.endTimestamp = RichPresence.endTimestamp; | ||||
|     rp.partySize = RichPresence.partySize; | ||||
|     rp.partyMax = RichPresence.partyMax; | ||||
|     rp.instance = RichPresence.instance; | ||||
|   | ||||
| @@ -11,7 +11,7 @@ | ||||
| * Ask to join callback data | ||||
| */ | ||||
| USTRUCT(BlueprintType) | ||||
| struct FDiscordJoinRequestData { | ||||
| struct FDiscordUserData { | ||||
|     GENERATED_USTRUCT_BODY() | ||||
|  | ||||
|     UPROPERTY(BlueprintReadOnly) | ||||
| @@ -24,15 +24,25 @@ 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") | ||||
| }; | ||||
|  | ||||
| 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 | ||||
|  | ||||
| @@ -89,7 +99,8 @@ public: | ||||
|               Category = "Discord") | ||||
|     void Initialize(const FString& applicationId, | ||||
|                     bool autoRegister, | ||||
|                     const FString& optionalSteamId); | ||||
|                     const FString& optionalSteamId, | ||||
|                     int optionalPipeNumber); | ||||
|  | ||||
|     UFUNCTION(BlueprintCallable, | ||||
|               meta = (DisplayName = "Shut down connection", Keywords = "Discord rpc"), | ||||
|   | ||||
| @@ -41,20 +41,20 @@ typedef struct DiscordRichPresence { | ||||
|     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 | ||||
| @@ -64,7 +64,8 @@ typedef struct DiscordEventHandlers { | ||||
| DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | ||||
|                                        DiscordEventHandlers* handlers, | ||||
|                                        int autoRegister, | ||||
|                                        const char* optionalSteamId); | ||||
|                                        const char* optionalSteamId, | ||||
|                                        int optionalPipeNumber); | ||||
| DISCORD_EXPORT void Discord_Shutdown(void); | ||||
|  | ||||
| /* checks for incoming messages, dispatches callbacks */ | ||||
| @@ -80,6 +81,8 @@ DISCORD_EXPORT void Discord_ClearPresence(void); | ||||
|  | ||||
| DISCORD_EXPORT void Discord_Respond(const char* userid, /* DISCORD_REPLY_ */ int reply); | ||||
|  | ||||
| DISCORD_EXPORT void Discord_UpdateHandlers(DiscordEventHandlers* handlers); | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| } /* extern "C" */ | ||||
| #endif | ||||
|   | ||||
| @@ -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) | ||||
|  | ||||
| @@ -55,7 +56,7 @@ if(WIN32) | ||||
|             /wd5027 # move assignment operator was implicitly defined as deleted | ||||
|         ) | ||||
|     endif(MSVC) | ||||
|     target_link_libraries(discord-rpc PRIVATE psapi) | ||||
|     target_link_libraries(discord-rpc PRIVATE psapi advapi32) | ||||
| endif(WIN32) | ||||
|  | ||||
| if(UNIX) | ||||
| @@ -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 | ||||
|   | ||||
| @@ -12,7 +12,7 @@ struct BaseConnection { | ||||
|     static BaseConnection* Create(); | ||||
|     static void Destroy(BaseConnection*&); | ||||
|     bool isOpen{false}; | ||||
|     bool Open(); | ||||
|     bool Open(int pipe); | ||||
|     bool Close(); | ||||
|     bool Write(const void* data, size_t length); | ||||
|     bool Read(void* data, size_t length); | ||||
|   | ||||
| @@ -49,7 +49,7 @@ static const char* GetTempPath() | ||||
|     c = nullptr; | ||||
| } | ||||
|  | ||||
| bool BaseConnection::Open() | ||||
| bool BaseConnection::Open(int pipe) | ||||
| { | ||||
|     const char* tempPath = GetTempPath(); | ||||
|     auto self = reinterpret_cast<BaseConnectionUnix*>(this); | ||||
| @@ -62,8 +62,7 @@ bool BaseConnection::Open() | ||||
|     int optval = 1; | ||||
|     setsockopt(self->sock, SOL_SOCKET, SO_NOSIGPIPE, &optval, sizeof(optval)); | ||||
| #endif | ||||
|  | ||||
|     for (int pipeNum = 0; pipeNum < 10; ++pipeNum) { | ||||
|     for (int pipeNum = pipe; pipeNum < 10; ++pipeNum) { | ||||
|         snprintf( | ||||
|           PipeAddr.sun_path, sizeof(PipeAddr.sun_path), "%s/discord-ipc-%d", tempPath, pipeNum); | ||||
|         int err = connect(self->sock, (const sockaddr*)&PipeAddr, sizeof(PipeAddr)); | ||||
| @@ -118,5 +117,8 @@ bool BaseConnection::Read(void* data, size_t length) | ||||
|         } | ||||
|         Close(); | ||||
|     } | ||||
|     else if (res == 0) { | ||||
|         Close(); | ||||
|     } | ||||
|     return res == (int)length; | ||||
| } | ||||
|   | ||||
| @@ -6,6 +6,7 @@ | ||||
| #define NOIME | ||||
| #include <assert.h> | ||||
| #include <windows.h> | ||||
| #include <sstream> | ||||
|  | ||||
| int GetProcessId() | ||||
| { | ||||
| @@ -30,11 +31,11 @@ static BaseConnectionWin Connection; | ||||
|     c = nullptr; | ||||
| } | ||||
|  | ||||
| bool BaseConnection::Open() | ||||
| bool BaseConnection::Open(int pipe) | ||||
| { | ||||
|     wchar_t pipeName[]{L"\\\\?\\pipe\\discord-ipc-0"}; | ||||
|     const size_t pipeDigit = sizeof(pipeName) / sizeof(wchar_t) - 2; | ||||
|     pipeName[pipeDigit] = L'0'; | ||||
|     pipeName[pipeDigit] += pipe; | ||||
|     auto self = reinterpret_cast<BaseConnectionWin*>(this); | ||||
|     for (;;) { | ||||
|         self->pipe = ::CreateFileW( | ||||
|   | ||||
| @@ -33,9 +33,11 @@ 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; | ||||
|     } | ||||
|  | ||||
| @@ -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); | ||||
|   | ||||
| @@ -26,7 +26,8 @@ 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); | ||||
| @@ -44,17 +45,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; | ||||
| } | ||||
|  | ||||
| @@ -138,7 +146,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}; | ||||
| @@ -60,9 +62,11 @@ static char LastErrorMessage[256]; | ||||
| static int LastDisconnectErrorCode{0}; | ||||
| static char LastDisconnectErrorMessage[256]; | ||||
| 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 | ||||
| @@ -86,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(); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
| @@ -115,7 +120,7 @@ public: | ||||
|     void Notify() {} | ||||
| }; | ||||
| #endif // DISCORD_DISABLE_IO_THREAD | ||||
| static IoThreadHolder IoThread; | ||||
| static IoThreadHolder* IoThread{nullptr}; | ||||
|  | ||||
| static void UpdateReconnectTime() | ||||
| { | ||||
| @@ -210,17 +215,17 @@ static void Discord_UpdateConnection(void) | ||||
|         } | ||||
|  | ||||
|         // writes | ||||
|         if (QueuedPresence.length) { | ||||
|         if (UpdatePresence.exchange(false) && QueuedPresence.length) { | ||||
|             QueuedMessage local; | ||||
|             PresenceMutex.lock(); | ||||
|             { | ||||
|                 std::lock_guard<std::mutex> guard(PresenceMutex); | ||||
|                 local.Copy(QueuedPresence); | ||||
|             QueuedPresence.length = 0; | ||||
|             PresenceMutex.unlock(); | ||||
|             } | ||||
|             if (!Connection->Write(local.buffer, local.length)) { | ||||
|                 // if we fail to send, requeue | ||||
|                 PresenceMutex.lock(); | ||||
|                 std::lock_guard<std::mutex> guard(PresenceMutex); | ||||
|                 QueuedPresence.Copy(local); | ||||
|                 PresenceMutex.unlock(); | ||||
|                 UpdatePresence.exchange(true); | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -234,7 +239,9 @@ static void Discord_UpdateConnection(void) | ||||
|  | ||||
| static void SignalIOActivity() | ||||
| { | ||||
|     IoThread.Notify(); | ||||
|     if (IoThread != nullptr) { | ||||
|         IoThread->Notify(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static bool RegisterForEvent(const char* evtName) | ||||
| @@ -250,11 +257,30 @@ static bool RegisterForEvent(const char* evtName) | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| static bool DeregisterForEvent(const char* evtName) | ||||
| { | ||||
|     auto qmessage = SendQueue.GetNextAddMessage(); | ||||
|     if (qmessage) { | ||||
|         qmessage->length = | ||||
|           JsonWriteUnsubscribeCommand(qmessage->buffer, sizeof(qmessage->buffer), Nonce++, evtName); | ||||
|         SendQueue.CommitAdd(); | ||||
|         SignalIOActivity(); | ||||
|         return true; | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| extern "C" DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | ||||
|                                                   DiscordEventHandlers* handlers, | ||||
|                                                   int autoRegister, | ||||
|                                                   const char* optionalSteamId) | ||||
|                                                   const char* optionalSteamId, | ||||
|                                                   int pipe) | ||||
| { | ||||
|     IoThread = new (std::nothrow) IoThreadHolder(); | ||||
|     if (IoThread == nullptr) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (autoRegister) { | ||||
|         if (optionalSteamId && optionalSteamId[0]) { | ||||
|             Discord_RegisterSteamGame(applicationId, optionalSteamId); | ||||
| @@ -266,10 +292,16 @@ extern "C" DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | ||||
|  | ||||
|     Pid = GetProcessId(); | ||||
|  | ||||
|     { | ||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||
|  | ||||
|         if (handlers) { | ||||
|         Handlers = *handlers; | ||||
|             QueuedHandlers = *handlers; | ||||
|         } | ||||
|         else { | ||||
|             QueuedHandlers = {}; | ||||
|         } | ||||
|  | ||||
|         Handlers = {}; | ||||
|     } | ||||
|  | ||||
| @@ -277,22 +309,34 @@ extern "C" DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     Connection = RpcConnection::Create(applicationId); | ||||
|     Connection->onConnect = []() { | ||||
|     Connection = RpcConnection::Create(applicationId, pipe); | ||||
|     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(); | ||||
|  | ||||
|         if (Handlers.joinGame) { | ||||
|             RegisterForEvent("ACTIVITY_JOIN"); | ||||
|         } | ||||
|  | ||||
|         if (Handlers.spectateGame) { | ||||
|             RegisterForEvent("ACTIVITY_SPECTATE"); | ||||
|         } | ||||
|  | ||||
|         if (Handlers.joinRequest) { | ||||
|             RegisterForEvent("ACTIVITY_JOIN_REQUEST"); | ||||
|         } | ||||
|     }; | ||||
|     Connection->onDisconnect = [](int err, const char* message) { | ||||
|         LastDisconnectErrorCode = err; | ||||
| @@ -301,7 +345,7 @@ extern "C" DISCORD_EXPORT void Discord_Initialize(const char* applicationId, | ||||
|         UpdateReconnectTime(); | ||||
|     }; | ||||
|  | ||||
|     IoThread.Start(); | ||||
|     IoThread->Start(); | ||||
| } | ||||
|  | ||||
| extern "C" DISCORD_EXPORT void Discord_Shutdown(void) | ||||
| @@ -312,16 +356,25 @@ 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); | ||||
| } | ||||
|  | ||||
| extern "C" DISCORD_EXPORT void Discord_UpdatePresence(const DiscordRichPresence* presence) | ||||
| { | ||||
|     PresenceMutex.lock(); | ||||
|     { | ||||
|         std::lock_guard<std::mutex> guard(PresenceMutex); | ||||
|         QueuedPresence.length = JsonWriteRichPresenceObj( | ||||
|           QueuedPresence.buffer, sizeof(QueuedPresence.buffer), Nonce++, Pid, presence); | ||||
|     PresenceMutex.unlock(); | ||||
|         UpdatePresence.exchange(true); | ||||
|     } | ||||
|     SignalIOActivity(); | ||||
| } | ||||
|  | ||||
| @@ -360,26 +413,43 @@ extern "C" DISCORD_EXPORT void Discord_RunCallbacks(void) | ||||
|  | ||||
|     if (isConnected) { | ||||
|         // if we are connected, disconnect cb first | ||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||
|         if (wasDisconnected && Handlers.disconnected) { | ||||
|             Handlers.disconnected(LastDisconnectErrorCode, LastDisconnectErrorMessage); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (WasJustConnected.exchange(false) && Handlers.ready) { | ||||
|         Handlers.ready(); | ||||
|     if (WasJustConnected.exchange(false)) { | ||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||
|         if (Handlers.ready) { | ||||
|             DiscordUser du{connectedUser.userId, | ||||
|                            connectedUser.username, | ||||
|                            connectedUser.discriminator, | ||||
|                            connectedUser.avatar}; | ||||
|             Handlers.ready(&du); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (GotErrorMessage.exchange(false) && Handlers.errored) { | ||||
|     if (GotErrorMessage.exchange(false)) { | ||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||
|         if (Handlers.errored) { | ||||
|             Handlers.errored(LastErrorCode, LastErrorMessage); | ||||
|         } | ||||
|  | ||||
|     if (WasJoinGame.exchange(false) && Handlers.joinGame) { | ||||
|         Handlers.joinGame(JoinGameSecret); | ||||
|     } | ||||
|  | ||||
|     if (WasSpectateGame.exchange(false) && Handlers.spectateGame) { | ||||
|     if (WasJoinGame.exchange(false)) { | ||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||
|         if (Handlers.joinGame) { | ||||
|             Handlers.joinGame(JoinGameSecret); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (WasSpectateGame.exchange(false)) { | ||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||
|         if (Handlers.spectateGame) { | ||||
|             Handlers.spectateGame(SpectateGameSecret); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Right now this batches up any requests and sends them all in a burst; I could imagine a world | ||||
|     // where the implementer would rather sequentially accept/reject each one before the next invite | ||||
| @@ -388,17 +458,48 @@ extern "C" DISCORD_EXPORT void Discord_RunCallbacks(void) | ||||
|     // not it should be trivial for the implementer to make a queue themselves. | ||||
|     while (JoinAskQueue.HavePendingSends()) { | ||||
|         auto req = JoinAskQueue.GetNextSendMessage(); | ||||
|         { | ||||
|             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(); | ||||
|     } | ||||
|  | ||||
|     if (!isConnected) { | ||||
|         // if we are not connected, disconnect message last | ||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||
|         if (wasDisconnected && Handlers.disconnected) { | ||||
|             Handlers.disconnected(LastDisconnectErrorCode, LastDisconnectErrorMessage); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| 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);                                    \ | ||||
|     }                                                               \ | ||||
|     else if (Handlers.handler_name && !newHandlers->handler_name) { \ | ||||
|         DeregisterForEvent(event);                                  \ | ||||
|     } | ||||
|  | ||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||
|         HANDLE_EVENT_REGISTRATION(joinGame, "ACTIVITY_JOIN") | ||||
|         HANDLE_EVENT_REGISTRATION(spectateGame, "ACTIVITY_SPECTATE") | ||||
|         HANDLE_EVENT_REGISTRATION(joinRequest, "ACTIVITY_JOIN_REQUEST") | ||||
|  | ||||
| #undef HANDLE_EVENT_REGISTRATION | ||||
|  | ||||
|         Handlers = *newHandlers; | ||||
|     } | ||||
|     else { | ||||
|         std::lock_guard<std::mutex> guard(HandlerMutex); | ||||
|         Handlers = {}; | ||||
|     } | ||||
|     return; | ||||
| } | ||||
|   | ||||
| @@ -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}; | ||||
|   | ||||
| @@ -6,10 +6,11 @@ | ||||
| static const int RpcVersion = 1; | ||||
| static RpcConnection Instance; | ||||
|  | ||||
| /*static*/ RpcConnection* RpcConnection::Create(const char* applicationId) | ||||
| /*static*/ RpcConnection* RpcConnection::Create(const char* applicationId, int pipe) | ||||
| { | ||||
|     Instance.connection = BaseConnection::Create(); | ||||
|     StringCopy(Instance.appId, applicationId); | ||||
|     Instance.pipe = pipe; | ||||
|     return &Instance; | ||||
| } | ||||
|  | ||||
| @@ -26,13 +27,9 @@ void RpcConnection::Open() | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (state == State::Disconnected) { | ||||
|         if (connection->Open()) { | ||||
|         } | ||||
|         else { | ||||
|     if (state == State::Disconnected && !connection->Open(Instance.pipe)) { | ||||
|         return; | ||||
|     } | ||||
|     } | ||||
|  | ||||
|     if (state == State::SentHandshake) { | ||||
|         JsonDocument message; | ||||
| @@ -42,7 +39,7 @@ void RpcConnection::Open() | ||||
|             if (cmd && evt && !strcmp(cmd, "DISPATCH") && !strcmp(evt, "READY")) { | ||||
|                 state = State::Connected; | ||||
|                 if (onConnect) { | ||||
|                     onConnect(); | ||||
|                     onConnect(message); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|   | ||||
| @@ -40,14 +40,15 @@ 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 pipe; | ||||
|     int lastErrorCode{0}; | ||||
|     char lastErrorMessage[256]{}; | ||||
|     RpcConnection::MessageFrame sendFrame; | ||||
|  | ||||
|     static RpcConnection* Create(const char* applicationId); | ||||
|     static RpcConnection* Create(const char* applicationId, int pipe); | ||||
|     static void Destroy(RpcConnection*&); | ||||
|  | ||||
|     inline bool IsOpen() const { return state == State::Connected; } | ||||
|   | ||||
| @@ -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); | ||||
| @@ -197,6 +196,25 @@ size_t JsonWriteSubscribeCommand(char* dest, size_t maxLen, int nonce, const cha | ||||
|     return writer.Size(); | ||||
| } | ||||
|  | ||||
| size_t JsonWriteUnsubscribeCommand(char* dest, size_t maxLen, int nonce, const char* evtName) | ||||
| { | ||||
|     JsonWriter writer(dest, maxLen); | ||||
|  | ||||
|     { | ||||
|         WriteObject obj(writer); | ||||
|  | ||||
|         JsonWriteNonce(writer, nonce); | ||||
|  | ||||
|         WriteKey(writer, "cmd"); | ||||
|         writer.String("UNSUBSCRIBE"); | ||||
|  | ||||
|         WriteKey(writer, "evt"); | ||||
|         writer.String(evtName); | ||||
|     } | ||||
|  | ||||
|     return writer.Size(); | ||||
| } | ||||
|  | ||||
| size_t JsonWriteJoinReply(char* dest, size_t maxLen, const char* userId, int reply, int nonce) | ||||
| { | ||||
|     JsonWriter writer(dest, maxLen); | ||||
|   | ||||
| @@ -47,6 +47,8 @@ size_t JsonWriteRichPresenceObj(char* dest, | ||||
|                                 const DiscordRichPresence* presence); | ||||
| size_t JsonWriteSubscribeCommand(char* dest, size_t maxLen, int nonce, const char* evtName); | ||||
|  | ||||
| size_t JsonWriteUnsubscribeCommand(char* dest, size_t maxLen, int nonce, const char* evtName); | ||||
|  | ||||
| size_t JsonWriteJoinReply(char* dest, size_t maxLen, const char* userId, int reply, int nonce); | ||||
|  | ||||
| // I want to use as few allocations as I can get away with, and to do that with RapidJson, you need | ||||
|   | ||||
		Reference in New Issue
	
	Block a user