// This file is provided under The MIT License as part of Steamworks.NET. // Copyright (c) 2013-2022 Riley Labrecque // Please see the included LICENSE.txt for additional information. // This file is automatically generated. // Changes to this file will be reverted when you update Steamworks.NET #if !(UNITY_STANDALONE_WIN || UNITY_STANDALONE_LINUX || UNITY_STANDALONE_OSX || STEAMWORKS_WIN || STEAMWORKS_LIN_OSX) #define DISABLESTEAMWORKS #endif #if !DISABLESTEAMWORKS using System.Runtime.InteropServices; using IntPtr = System.IntPtr; namespace Steamworks { public static class SteamMatchmaking { /// /// game server favorites storage /// saves basic details about a multiplayer game server locally /// returns the number of favorites servers the user has stored /// public static int GetFavoriteGameCount() { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_GetFavoriteGameCount(CSteamAPIContext.GetSteamMatchmaking()); } /// /// returns the details of the game server /// iGame is of range [0,GetFavoriteGameCount()) /// *pnIP, *pnConnPort are filled in the with IP:port of the game server /// *punFlags specify whether the game server was stored as an explicit favorite or in the history of connections /// *pRTime32LastPlayedOnServer is filled in the with the Unix time the favorite was added /// public static bool GetFavoriteGame(int iGame, out AppId_t pnAppID, out uint pnIP, out ushort pnConnPort, out ushort pnQueryPort, out uint punFlags, out uint pRTime32LastPlayedOnServer) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_GetFavoriteGame(CSteamAPIContext.GetSteamMatchmaking(), iGame, out pnAppID, out pnIP, out pnConnPort, out pnQueryPort, out punFlags, out pRTime32LastPlayedOnServer); } /// /// adds the game server to the local list; updates the time played of the server if it already exists in the list /// public static int AddFavoriteGame(AppId_t nAppID, uint nIP, ushort nConnPort, ushort nQueryPort, uint unFlags, uint rTime32LastPlayedOnServer) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_AddFavoriteGame(CSteamAPIContext.GetSteamMatchmaking(), nAppID, nIP, nConnPort, nQueryPort, unFlags, rTime32LastPlayedOnServer); } /// /// removes the game server from the local storage; returns true if one was removed /// public static bool RemoveFavoriteGame(AppId_t nAppID, uint nIP, ushort nConnPort, ushort nQueryPort, uint unFlags) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_RemoveFavoriteGame(CSteamAPIContext.GetSteamMatchmaking(), nAppID, nIP, nConnPort, nQueryPort, unFlags); } /// /// ///// /// Game lobby functions /// Get a list of relevant lobbies /// this is an asynchronous request /// results will be returned by LobbyMatchList_t callback & call result, with the number of lobbies found /// this will never return lobbies that are full /// to add more filter, the filter calls below need to be call before each and every RequestLobbyList() call /// use the CCallResult<> object in steam_api.h to match the SteamAPICall_t call result to a function in an object, e.g. /// class CMyLobbyListManager /// { /// CCallResult<CMyLobbyListManager, LobbyMatchList_t> m_CallResultLobbyMatchList; /// void FindLobbies() /// { /// // SteamMatchmaking()->AddRequestLobbyListFilter*() functions would be called here, before RequestLobbyList() /// SteamAPICall_t hSteamAPICall = SteamMatchmaking()->RequestLobbyList(); /// m_CallResultLobbyMatchList.Set( hSteamAPICall, this, &CMyLobbyListManager::OnLobbyMatchList ); /// } /// void OnLobbyMatchList( LobbyMatchList_t *pLobbyMatchList, bool bIOFailure ) /// { /// // lobby list has be retrieved from Steam back-end, use results /// } /// } /// public static SteamAPICall_t RequestLobbyList() { InteropHelp.TestIfAvailableClient(); return (SteamAPICall_t)NativeMethods.ISteamMatchmaking_RequestLobbyList(CSteamAPIContext.GetSteamMatchmaking()); } /// /// filters for lobbies /// this needs to be called before RequestLobbyList() to take effect /// these are cleared on each call to RequestLobbyList() /// public static void AddRequestLobbyListStringFilter(string pchKeyToMatch, string pchValueToMatch, ELobbyComparison eComparisonType) { InteropHelp.TestIfAvailableClient(); using (var pchKeyToMatch2 = new InteropHelp.UTF8StringHandle(pchKeyToMatch)) using (var pchValueToMatch2 = new InteropHelp.UTF8StringHandle(pchValueToMatch)) { NativeMethods.ISteamMatchmaking_AddRequestLobbyListStringFilter(CSteamAPIContext.GetSteamMatchmaking(), pchKeyToMatch2, pchValueToMatch2, eComparisonType); } } /// /// numerical comparison /// public static void AddRequestLobbyListNumericalFilter(string pchKeyToMatch, int nValueToMatch, ELobbyComparison eComparisonType) { InteropHelp.TestIfAvailableClient(); using (var pchKeyToMatch2 = new InteropHelp.UTF8StringHandle(pchKeyToMatch)) { NativeMethods.ISteamMatchmaking_AddRequestLobbyListNumericalFilter(CSteamAPIContext.GetSteamMatchmaking(), pchKeyToMatch2, nValueToMatch, eComparisonType); } } /// /// returns results closest to the specified value. Multiple near filters can be added, with early filters taking precedence /// public static void AddRequestLobbyListNearValueFilter(string pchKeyToMatch, int nValueToBeCloseTo) { InteropHelp.TestIfAvailableClient(); using (var pchKeyToMatch2 = new InteropHelp.UTF8StringHandle(pchKeyToMatch)) { NativeMethods.ISteamMatchmaking_AddRequestLobbyListNearValueFilter(CSteamAPIContext.GetSteamMatchmaking(), pchKeyToMatch2, nValueToBeCloseTo); } } /// /// returns only lobbies with the specified number of slots available /// public static void AddRequestLobbyListFilterSlotsAvailable(int nSlotsAvailable) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmaking_AddRequestLobbyListFilterSlotsAvailable(CSteamAPIContext.GetSteamMatchmaking(), nSlotsAvailable); } /// /// sets the distance for which we should search for lobbies (based on users IP address to location map on the Steam backed) /// public static void AddRequestLobbyListDistanceFilter(ELobbyDistanceFilter eLobbyDistanceFilter) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmaking_AddRequestLobbyListDistanceFilter(CSteamAPIContext.GetSteamMatchmaking(), eLobbyDistanceFilter); } /// /// sets how many results to return, the lower the count the faster it is to download the lobby results & details to the client /// public static void AddRequestLobbyListResultCountFilter(int cMaxResults) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmaking_AddRequestLobbyListResultCountFilter(CSteamAPIContext.GetSteamMatchmaking(), cMaxResults); } public static void AddRequestLobbyListCompatibleMembersFilter(CSteamID steamIDLobby) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmaking_AddRequestLobbyListCompatibleMembersFilter(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby); } /// /// returns the CSteamID of a lobby, as retrieved by a RequestLobbyList call /// should only be called after a LobbyMatchList_t callback is received /// iLobby is of the range [0, LobbyMatchList_t::m_nLobbiesMatching) /// the returned CSteamID::IsValid() will be false if iLobby is out of range /// public static CSteamID GetLobbyByIndex(int iLobby) { InteropHelp.TestIfAvailableClient(); return (CSteamID)NativeMethods.ISteamMatchmaking_GetLobbyByIndex(CSteamAPIContext.GetSteamMatchmaking(), iLobby); } /// /// Create a lobby on the Steam servers. /// If private, then the lobby will not be returned by any RequestLobbyList() call; the CSteamID /// of the lobby will need to be communicated via game channels or via InviteUserToLobby() /// this is an asynchronous request /// results will be returned by LobbyCreated_t callback and call result; lobby is joined & ready to use at this point /// a LobbyEnter_t callback will also be received (since the local user is joining their own lobby) /// public static SteamAPICall_t CreateLobby(ELobbyType eLobbyType, int cMaxMembers) { InteropHelp.TestIfAvailableClient(); return (SteamAPICall_t)NativeMethods.ISteamMatchmaking_CreateLobby(CSteamAPIContext.GetSteamMatchmaking(), eLobbyType, cMaxMembers); } /// /// Joins an existing lobby /// this is an asynchronous request /// results will be returned by LobbyEnter_t callback & call result, check m_EChatRoomEnterResponse to see if was successful /// lobby metadata is available to use immediately on this call completing /// public static SteamAPICall_t JoinLobby(CSteamID steamIDLobby) { InteropHelp.TestIfAvailableClient(); return (SteamAPICall_t)NativeMethods.ISteamMatchmaking_JoinLobby(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby); } /// /// Leave a lobby; this will take effect immediately on the client side /// other users in the lobby will be notified by a LobbyChatUpdate_t callback /// public static void LeaveLobby(CSteamID steamIDLobby) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmaking_LeaveLobby(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby); } /// /// Invite another user to the lobby /// the target user will receive a LobbyInvite_t callback /// will return true if the invite is successfully sent, whether or not the target responds /// returns false if the local user is not connected to the Steam servers /// if the other user clicks the join link, a GameLobbyJoinRequested_t will be posted if the user is in-game, /// or if the game isn't running yet the game will be launched with the parameter +connect_lobby <64-bit lobby id> /// public static bool InviteUserToLobby(CSteamID steamIDLobby, CSteamID steamIDInvitee) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_InviteUserToLobby(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, steamIDInvitee); } /// /// Lobby iteration, for viewing details of users in a lobby /// only accessible if the lobby user is a member of the specified lobby /// persona information for other lobby members (name, avatar, etc.) will be asynchronously received /// and accessible via ISteamFriends interface /// returns the number of users in the specified lobby /// public static int GetNumLobbyMembers(CSteamID steamIDLobby) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_GetNumLobbyMembers(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby); } /// /// returns the CSteamID of a user in the lobby /// iMember is of range [0,GetNumLobbyMembers()) /// note that the current user must be in a lobby to retrieve CSteamIDs of other users in that lobby /// public static CSteamID GetLobbyMemberByIndex(CSteamID steamIDLobby, int iMember) { InteropHelp.TestIfAvailableClient(); return (CSteamID)NativeMethods.ISteamMatchmaking_GetLobbyMemberByIndex(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, iMember); } /// /// Get data associated with this lobby /// takes a simple key, and returns the string associated with it /// "" will be returned if no value is set, or if steamIDLobby is invalid /// public static string GetLobbyData(CSteamID steamIDLobby, string pchKey) { InteropHelp.TestIfAvailableClient(); using (var pchKey2 = new InteropHelp.UTF8StringHandle(pchKey)) { return InteropHelp.PtrToStringUTF8(NativeMethods.ISteamMatchmaking_GetLobbyData(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, pchKey2)); } } /// /// Sets a key/value pair in the lobby metadata /// each user in the lobby will be broadcast this new value, and any new users joining will receive any existing data /// this can be used to set lobby names, map, etc. /// to reset a key, just set it to "" /// other users in the lobby will receive notification of the lobby data change via a LobbyDataUpdate_t callback /// public static bool SetLobbyData(CSteamID steamIDLobby, string pchKey, string pchValue) { InteropHelp.TestIfAvailableClient(); using (var pchKey2 = new InteropHelp.UTF8StringHandle(pchKey)) using (var pchValue2 = new InteropHelp.UTF8StringHandle(pchValue)) { return NativeMethods.ISteamMatchmaking_SetLobbyData(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, pchKey2, pchValue2); } } /// /// returns the number of metadata keys set on the specified lobby /// public static int GetLobbyDataCount(CSteamID steamIDLobby) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_GetLobbyDataCount(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby); } /// /// returns a lobby metadata key/values pair by index, of range [0, GetLobbyDataCount()) /// public static bool GetLobbyDataByIndex(CSteamID steamIDLobby, int iLobbyData, out string pchKey, int cchKeyBufferSize, out string pchValue, int cchValueBufferSize) { InteropHelp.TestIfAvailableClient(); IntPtr pchKey2 = Marshal.AllocHGlobal(cchKeyBufferSize); IntPtr pchValue2 = Marshal.AllocHGlobal(cchValueBufferSize); bool ret = NativeMethods.ISteamMatchmaking_GetLobbyDataByIndex(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, iLobbyData, pchKey2, cchKeyBufferSize, pchValue2, cchValueBufferSize); pchKey = ret ? InteropHelp.PtrToStringUTF8(pchKey2) : null; Marshal.FreeHGlobal(pchKey2); pchValue = ret ? InteropHelp.PtrToStringUTF8(pchValue2) : null; Marshal.FreeHGlobal(pchValue2); return ret; } /// /// removes a metadata key from the lobby /// public static bool DeleteLobbyData(CSteamID steamIDLobby, string pchKey) { InteropHelp.TestIfAvailableClient(); using (var pchKey2 = new InteropHelp.UTF8StringHandle(pchKey)) { return NativeMethods.ISteamMatchmaking_DeleteLobbyData(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, pchKey2); } } /// /// Gets per-user metadata for someone in this lobby /// public static string GetLobbyMemberData(CSteamID steamIDLobby, CSteamID steamIDUser, string pchKey) { InteropHelp.TestIfAvailableClient(); using (var pchKey2 = new InteropHelp.UTF8StringHandle(pchKey)) { return InteropHelp.PtrToStringUTF8(NativeMethods.ISteamMatchmaking_GetLobbyMemberData(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, steamIDUser, pchKey2)); } } /// /// Sets per-user metadata (for the local user implicitly) /// public static void SetLobbyMemberData(CSteamID steamIDLobby, string pchKey, string pchValue) { InteropHelp.TestIfAvailableClient(); using (var pchKey2 = new InteropHelp.UTF8StringHandle(pchKey)) using (var pchValue2 = new InteropHelp.UTF8StringHandle(pchValue)) { NativeMethods.ISteamMatchmaking_SetLobbyMemberData(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, pchKey2, pchValue2); } } /// /// Broadcasts a chat message to the all the users in the lobby /// users in the lobby (including the local user) will receive a LobbyChatMsg_t callback /// returns true if the message is successfully sent /// pvMsgBody can be binary or text data, up to 4k /// if pvMsgBody is text, cubMsgBody should be strlen( text ) + 1, to include the null terminator /// public static bool SendLobbyChatMsg(CSteamID steamIDLobby, byte[] pvMsgBody, int cubMsgBody) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_SendLobbyChatMsg(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, pvMsgBody, cubMsgBody); } /// /// Get a chat message as specified in a LobbyChatMsg_t callback /// iChatID is the LobbyChatMsg_t::m_iChatID value in the callback /// *pSteamIDUser is filled in with the CSteamID of the member /// *pvData is filled in with the message itself /// return value is the number of bytes written into the buffer /// public static int GetLobbyChatEntry(CSteamID steamIDLobby, int iChatID, out CSteamID pSteamIDUser, byte[] pvData, int cubData, out EChatEntryType peChatEntryType) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_GetLobbyChatEntry(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, iChatID, out pSteamIDUser, pvData, cubData, out peChatEntryType); } /// /// Refreshes metadata for a lobby you're not necessarily in right now /// you never do this for lobbies you're a member of, only if your /// this will send down all the metadata associated with a lobby /// this is an asynchronous call /// returns false if the local user is not connected to the Steam servers /// results will be returned by a LobbyDataUpdate_t callback /// if the specified lobby doesn't exist, LobbyDataUpdate_t::m_bSuccess will be set to false /// public static bool RequestLobbyData(CSteamID steamIDLobby) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_RequestLobbyData(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby); } /// /// sets the game server associated with the lobby /// usually at this point, the users will join the specified game server /// either the IP/Port or the steamID of the game server has to be valid, depending on how you want the clients to be able to connect /// public static void SetLobbyGameServer(CSteamID steamIDLobby, uint unGameServerIP, ushort unGameServerPort, CSteamID steamIDGameServer) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmaking_SetLobbyGameServer(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, unGameServerIP, unGameServerPort, steamIDGameServer); } /// /// returns the details of a game server set in a lobby - returns false if there is no game server set, or that lobby doesn't exist /// public static bool GetLobbyGameServer(CSteamID steamIDLobby, out uint punGameServerIP, out ushort punGameServerPort, out CSteamID psteamIDGameServer) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_GetLobbyGameServer(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, out punGameServerIP, out punGameServerPort, out psteamIDGameServer); } /// /// set the limit on the # of users who can join the lobby /// public static bool SetLobbyMemberLimit(CSteamID steamIDLobby, int cMaxMembers) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_SetLobbyMemberLimit(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, cMaxMembers); } /// /// returns the current limit on the # of users who can join the lobby; returns 0 if no limit is defined /// public static int GetLobbyMemberLimit(CSteamID steamIDLobby) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_GetLobbyMemberLimit(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby); } /// /// updates which type of lobby it is /// only lobbies that are k_ELobbyTypePublic or k_ELobbyTypeInvisible, and are set to joinable, will be returned by RequestLobbyList() calls /// public static bool SetLobbyType(CSteamID steamIDLobby, ELobbyType eLobbyType) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_SetLobbyType(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, eLobbyType); } /// /// sets whether or not a lobby is joinable - defaults to true for a new lobby /// if set to false, no user can join, even if they are a friend or have been invited /// public static bool SetLobbyJoinable(CSteamID steamIDLobby, bool bLobbyJoinable) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_SetLobbyJoinable(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, bLobbyJoinable); } /// /// returns the current lobby owner /// you must be a member of the lobby to access this /// there always one lobby owner - if the current owner leaves, another user will become the owner /// it is possible (bur rare) to join a lobby just as the owner is leaving, thus entering a lobby with self as the owner /// public static CSteamID GetLobbyOwner(CSteamID steamIDLobby) { InteropHelp.TestIfAvailableClient(); return (CSteamID)NativeMethods.ISteamMatchmaking_GetLobbyOwner(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby); } /// /// changes who the lobby owner is /// you must be the lobby owner for this to succeed, and steamIDNewOwner must be in the lobby /// after completion, the local user will no longer be the owner /// public static bool SetLobbyOwner(CSteamID steamIDLobby, CSteamID steamIDNewOwner) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_SetLobbyOwner(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, steamIDNewOwner); } /// /// link two lobbies for the purposes of checking player compatibility /// you must be the lobby owner of both lobbies /// public static bool SetLinkedLobby(CSteamID steamIDLobby, CSteamID steamIDLobbyDependent) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmaking_SetLinkedLobby(CSteamAPIContext.GetSteamMatchmaking(), steamIDLobby, steamIDLobbyDependent); } #if _PS3 /// /// changes who the lobby owner is /// you must be the lobby owner for this to succeed, and steamIDNewOwner must be in the lobby /// after completion, the local user will no longer be the owner /// public static void CheckForPSNGameBootInvite(uint iGameBootAttributes) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmaking_CheckForPSNGameBootInvite(CSteamAPIContext.GetSteamMatchmaking(), iGameBootAttributes); } #endif } public static class SteamMatchmakingServers { /// /// Request a new list of servers of a particular type. These calls each correspond to one of the EMatchMakingType values. /// Each call allocates a new asynchronous request object. /// Request object must be released by calling ReleaseRequest( hServerListRequest ) /// public static HServerListRequest RequestInternetServerList(AppId_t iApp, MatchMakingKeyValuePair_t[] ppchFilters, uint nFilters, ISteamMatchmakingServerListResponse pRequestServersResponse) { InteropHelp.TestIfAvailableClient(); return (HServerListRequest)NativeMethods.ISteamMatchmakingServers_RequestInternetServerList(CSteamAPIContext.GetSteamMatchmakingServers(), iApp, new MMKVPMarshaller(ppchFilters), nFilters, (IntPtr)pRequestServersResponse); } public static HServerListRequest RequestLANServerList(AppId_t iApp, ISteamMatchmakingServerListResponse pRequestServersResponse) { InteropHelp.TestIfAvailableClient(); return (HServerListRequest)NativeMethods.ISteamMatchmakingServers_RequestLANServerList(CSteamAPIContext.GetSteamMatchmakingServers(), iApp, (IntPtr)pRequestServersResponse); } public static HServerListRequest RequestFriendsServerList(AppId_t iApp, MatchMakingKeyValuePair_t[] ppchFilters, uint nFilters, ISteamMatchmakingServerListResponse pRequestServersResponse) { InteropHelp.TestIfAvailableClient(); return (HServerListRequest)NativeMethods.ISteamMatchmakingServers_RequestFriendsServerList(CSteamAPIContext.GetSteamMatchmakingServers(), iApp, new MMKVPMarshaller(ppchFilters), nFilters, (IntPtr)pRequestServersResponse); } public static HServerListRequest RequestFavoritesServerList(AppId_t iApp, MatchMakingKeyValuePair_t[] ppchFilters, uint nFilters, ISteamMatchmakingServerListResponse pRequestServersResponse) { InteropHelp.TestIfAvailableClient(); return (HServerListRequest)NativeMethods.ISteamMatchmakingServers_RequestFavoritesServerList(CSteamAPIContext.GetSteamMatchmakingServers(), iApp, new MMKVPMarshaller(ppchFilters), nFilters, (IntPtr)pRequestServersResponse); } public static HServerListRequest RequestHistoryServerList(AppId_t iApp, MatchMakingKeyValuePair_t[] ppchFilters, uint nFilters, ISteamMatchmakingServerListResponse pRequestServersResponse) { InteropHelp.TestIfAvailableClient(); return (HServerListRequest)NativeMethods.ISteamMatchmakingServers_RequestHistoryServerList(CSteamAPIContext.GetSteamMatchmakingServers(), iApp, new MMKVPMarshaller(ppchFilters), nFilters, (IntPtr)pRequestServersResponse); } public static HServerListRequest RequestSpectatorServerList(AppId_t iApp, MatchMakingKeyValuePair_t[] ppchFilters, uint nFilters, ISteamMatchmakingServerListResponse pRequestServersResponse) { InteropHelp.TestIfAvailableClient(); return (HServerListRequest)NativeMethods.ISteamMatchmakingServers_RequestSpectatorServerList(CSteamAPIContext.GetSteamMatchmakingServers(), iApp, new MMKVPMarshaller(ppchFilters), nFilters, (IntPtr)pRequestServersResponse); } /// /// Releases the asynchronous request object and cancels any pending query on it if there's a pending query in progress. /// RefreshComplete callback is not posted when request is released. /// public static void ReleaseRequest(HServerListRequest hServerListRequest) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmakingServers_ReleaseRequest(CSteamAPIContext.GetSteamMatchmakingServers(), hServerListRequest); } /// /// the filter operation codes that go in the key part of MatchMakingKeyValuePair_t should be one of these: /// "map" /// - Server passes the filter if the server is playing the specified map. /// "gamedataand" /// - Server passes the filter if the server's game data (ISteamGameServer::SetGameData) contains all of the /// specified strings. The value field is a comma-delimited list of strings to match. /// "gamedataor" /// - Server passes the filter if the server's game data (ISteamGameServer::SetGameData) contains at least one of the /// specified strings. The value field is a comma-delimited list of strings to match. /// "gamedatanor" /// - Server passes the filter if the server's game data (ISteamGameServer::SetGameData) does not contain any /// of the specified strings. The value field is a comma-delimited list of strings to check. /// "gametagsand" /// - Server passes the filter if the server's game tags (ISteamGameServer::SetGameTags) contains all /// of the specified strings. The value field is a comma-delimited list of strings to check. /// "gametagsnor" /// - Server passes the filter if the server's game tags (ISteamGameServer::SetGameTags) does not contain any /// of the specified strings. The value field is a comma-delimited list of strings to check. /// "and" (x1 && x2 && ... && xn) /// "or" (x1 || x2 || ... || xn) /// "nand" !(x1 && x2 && ... && xn) /// "nor" !(x1 || x2 || ... || xn) /// - Performs Boolean operation on the following filters. The operand to this filter specifies /// the "size" of the Boolean inputs to the operation, in Key/value pairs. (The keyvalue /// pairs must immediately follow, i.e. this is a prefix logical operator notation.) /// In the simplest case where Boolean expressions are not nested, this is simply /// the number of operands. /// For example, to match servers on a particular map or with a particular tag, would would /// use these filters. /// ( server.map == "cp_dustbowl" || server.gametags.contains("payload") ) /// "or", "2" /// "map", "cp_dustbowl" /// "gametagsand", "payload" /// If logical inputs are nested, then the operand specifies the size of the entire /// "length" of its operands, not the number of immediate children. /// ( server.map == "cp_dustbowl" || ( server.gametags.contains("payload") && !server.gametags.contains("payloadrace") ) ) /// "or", "4" /// "map", "cp_dustbowl" /// "and", "2" /// "gametagsand", "payload" /// "gametagsnor", "payloadrace" /// Unary NOT can be achieved using either "nand" or "nor" with a single operand. /// "addr" /// - Server passes the filter if the server's query address matches the specified IP or IP:port. /// "gameaddr" /// - Server passes the filter if the server's game address matches the specified IP or IP:port. /// The following filter operations ignore the "value" part of MatchMakingKeyValuePair_t /// "dedicated" /// - Server passes the filter if it passed true to SetDedicatedServer. /// "secure" /// - Server passes the filter if the server is VAC-enabled. /// "notfull" /// - Server passes the filter if the player count is less than the reported max player count. /// "hasplayers" /// - Server passes the filter if the player count is greater than zero. /// "noplayers" /// - Server passes the filter if it doesn't have any players. /// "linux" /// - Server passes the filter if it's a linux server /// Get details on a given server in the list, you can get the valid range of index /// values by calling GetServerCount(). You will also receive index values in /// ISteamMatchmakingServerListResponse::ServerResponded() callbacks /// public static gameserveritem_t GetServerDetails(HServerListRequest hRequest, int iServer) { InteropHelp.TestIfAvailableClient(); return (gameserveritem_t)Marshal.PtrToStructure(NativeMethods.ISteamMatchmakingServers_GetServerDetails(CSteamAPIContext.GetSteamMatchmakingServers(), hRequest, iServer), typeof(gameserveritem_t)); } /// /// Cancel an request which is operation on the given list type. You should call this to cancel /// any in-progress requests before destructing a callback object that may have been passed /// to one of the above list request calls. Not doing so may result in a crash when a callback /// occurs on the destructed object. /// Canceling a query does not release the allocated request handle. /// The request handle must be released using ReleaseRequest( hRequest ) /// public static void CancelQuery(HServerListRequest hRequest) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmakingServers_CancelQuery(CSteamAPIContext.GetSteamMatchmakingServers(), hRequest); } /// /// Ping every server in your list again but don't update the list of servers /// Query callback installed when the server list was requested will be used /// again to post notifications and RefreshComplete, so the callback must remain /// valid until another RefreshComplete is called on it or the request /// is released with ReleaseRequest( hRequest ) /// public static void RefreshQuery(HServerListRequest hRequest) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmakingServers_RefreshQuery(CSteamAPIContext.GetSteamMatchmakingServers(), hRequest); } /// /// Returns true if the list is currently refreshing its server list /// public static bool IsRefreshing(HServerListRequest hRequest) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmakingServers_IsRefreshing(CSteamAPIContext.GetSteamMatchmakingServers(), hRequest); } /// /// How many servers in the given list, GetServerDetails above takes 0... GetServerCount() - 1 /// public static int GetServerCount(HServerListRequest hRequest) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamMatchmakingServers_GetServerCount(CSteamAPIContext.GetSteamMatchmakingServers(), hRequest); } /// /// Refresh a single server inside of a query (rather than all the servers ) /// public static void RefreshServer(HServerListRequest hRequest, int iServer) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmakingServers_RefreshServer(CSteamAPIContext.GetSteamMatchmakingServers(), hRequest, iServer); } /// /// ----------------------------------------------------------------------------- /// Queries to individual servers directly via IP/Port /// ----------------------------------------------------------------------------- /// Request updated ping time and other details from a single server /// public static HServerQuery PingServer(uint unIP, ushort usPort, ISteamMatchmakingPingResponse pRequestServersResponse) { InteropHelp.TestIfAvailableClient(); return (HServerQuery)NativeMethods.ISteamMatchmakingServers_PingServer(CSteamAPIContext.GetSteamMatchmakingServers(), unIP, usPort, (IntPtr)pRequestServersResponse); } /// /// Request the list of players currently playing on a server /// public static HServerQuery PlayerDetails(uint unIP, ushort usPort, ISteamMatchmakingPlayersResponse pRequestServersResponse) { InteropHelp.TestIfAvailableClient(); return (HServerQuery)NativeMethods.ISteamMatchmakingServers_PlayerDetails(CSteamAPIContext.GetSteamMatchmakingServers(), unIP, usPort, (IntPtr)pRequestServersResponse); } /// /// Request the list of rules that the server is running (See ISteamGameServer::SetKeyValue() to set the rules server side) /// public static HServerQuery ServerRules(uint unIP, ushort usPort, ISteamMatchmakingRulesResponse pRequestServersResponse) { InteropHelp.TestIfAvailableClient(); return (HServerQuery)NativeMethods.ISteamMatchmakingServers_ServerRules(CSteamAPIContext.GetSteamMatchmakingServers(), unIP, usPort, (IntPtr)pRequestServersResponse); } /// /// Cancel an outstanding Ping/Players/Rules query from above. You should call this to cancel /// any in-progress requests before destructing a callback object that may have been passed /// to one of the above calls to avoid crashing when callbacks occur. /// public static void CancelServerQuery(HServerQuery hServerQuery) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamMatchmakingServers_CancelServerQuery(CSteamAPIContext.GetSteamMatchmakingServers(), hServerQuery); } } public static class SteamGameSearch { /// /// ============================================================================================= /// Game Player APIs /// a keyname and a list of comma separated values: one of which is must be found in order for the match to qualify /// fails if a search is currently in progress /// public static EGameSearchErrorCode_t AddGameSearchParams(string pchKeyToFind, string pchValuesToFind) { InteropHelp.TestIfAvailableClient(); using (var pchKeyToFind2 = new InteropHelp.UTF8StringHandle(pchKeyToFind)) using (var pchValuesToFind2 = new InteropHelp.UTF8StringHandle(pchValuesToFind)) { return NativeMethods.ISteamGameSearch_AddGameSearchParams(CSteamAPIContext.GetSteamGameSearch(), pchKeyToFind2, pchValuesToFind2); } } /// /// all players in lobby enter the queue and await a SearchForGameNotificationCallback_t callback. fails if another search is currently in progress /// if not the owner of the lobby or search already in progress this call fails /// periodic callbacks will be sent as queue time estimates change /// public static EGameSearchErrorCode_t SearchForGameWithLobby(CSteamID steamIDLobby, int nPlayerMin, int nPlayerMax) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamGameSearch_SearchForGameWithLobby(CSteamAPIContext.GetSteamGameSearch(), steamIDLobby, nPlayerMin, nPlayerMax); } /// /// user enter the queue and await a SearchForGameNotificationCallback_t callback. fails if another search is currently in progress /// periodic callbacks will be sent as queue time estimates change /// public static EGameSearchErrorCode_t SearchForGameSolo(int nPlayerMin, int nPlayerMax) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamGameSearch_SearchForGameSolo(CSteamAPIContext.GetSteamGameSearch(), nPlayerMin, nPlayerMax); } /// /// after receiving SearchForGameResultCallback_t, accept or decline the game /// multiple SearchForGameResultCallback_t will follow as players accept game until the host starts or cancels the game /// public static EGameSearchErrorCode_t AcceptGame() { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamGameSearch_AcceptGame(CSteamAPIContext.GetSteamGameSearch()); } public static EGameSearchErrorCode_t DeclineGame() { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamGameSearch_DeclineGame(CSteamAPIContext.GetSteamGameSearch()); } /// /// after receiving GameStartedByHostCallback_t get connection details to server /// public static EGameSearchErrorCode_t RetrieveConnectionDetails(CSteamID steamIDHost, out string pchConnectionDetails, int cubConnectionDetails) { InteropHelp.TestIfAvailableClient(); IntPtr pchConnectionDetails2 = Marshal.AllocHGlobal(cubConnectionDetails); EGameSearchErrorCode_t ret = NativeMethods.ISteamGameSearch_RetrieveConnectionDetails(CSteamAPIContext.GetSteamGameSearch(), steamIDHost, pchConnectionDetails2, cubConnectionDetails); pchConnectionDetails = ret != 0 ? InteropHelp.PtrToStringUTF8(pchConnectionDetails2) : null; Marshal.FreeHGlobal(pchConnectionDetails2); return ret; } /// /// leaves queue if still waiting /// public static EGameSearchErrorCode_t EndGameSearch() { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamGameSearch_EndGameSearch(CSteamAPIContext.GetSteamGameSearch()); } /// /// ============================================================================================= /// Game Host APIs /// a keyname and a list of comma separated values: all the values you allow /// public static EGameSearchErrorCode_t SetGameHostParams(string pchKey, string pchValue) { InteropHelp.TestIfAvailableClient(); using (var pchKey2 = new InteropHelp.UTF8StringHandle(pchKey)) using (var pchValue2 = new InteropHelp.UTF8StringHandle(pchValue)) { return NativeMethods.ISteamGameSearch_SetGameHostParams(CSteamAPIContext.GetSteamGameSearch(), pchKey2, pchValue2); } } /// /// set connection details for players once game is found so they can connect to this server /// public static EGameSearchErrorCode_t SetConnectionDetails(string pchConnectionDetails, int cubConnectionDetails) { InteropHelp.TestIfAvailableClient(); using (var pchConnectionDetails2 = new InteropHelp.UTF8StringHandle(pchConnectionDetails)) { return NativeMethods.ISteamGameSearch_SetConnectionDetails(CSteamAPIContext.GetSteamGameSearch(), pchConnectionDetails2, cubConnectionDetails); } } /// /// mark server as available for more players with nPlayerMin,nPlayerMax desired /// accept no lobbies with playercount greater than nMaxTeamSize /// the set of lobbies returned must be partitionable into teams of no more than nMaxTeamSize /// RequestPlayersForGameNotificationCallback_t callback will be sent when the search has started /// multple RequestPlayersForGameResultCallback_t callbacks will follow when players are found /// public static EGameSearchErrorCode_t RequestPlayersForGame(int nPlayerMin, int nPlayerMax, int nMaxTeamSize) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamGameSearch_RequestPlayersForGame(CSteamAPIContext.GetSteamGameSearch(), nPlayerMin, nPlayerMax, nMaxTeamSize); } /// /// accept the player list and release connection details to players /// players will only be given connection details and host steamid when this is called /// ( allows host to accept after all players confirm, some confirm, or none confirm. decision is entirely up to the host ) /// public static EGameSearchErrorCode_t HostConfirmGameStart(ulong ullUniqueGameID) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamGameSearch_HostConfirmGameStart(CSteamAPIContext.GetSteamGameSearch(), ullUniqueGameID); } /// /// cancel request and leave the pool of game hosts looking for players /// if a set of players has already been sent to host, all players will receive SearchForGameHostFailedToConfirm_t /// public static EGameSearchErrorCode_t CancelRequestPlayersForGame() { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamGameSearch_CancelRequestPlayersForGame(CSteamAPIContext.GetSteamGameSearch()); } /// /// submit a result for one player. does not end the game. ullUniqueGameID continues to describe this game /// public static EGameSearchErrorCode_t SubmitPlayerResult(ulong ullUniqueGameID, CSteamID steamIDPlayer, EPlayerResult_t EPlayerResult) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamGameSearch_SubmitPlayerResult(CSteamAPIContext.GetSteamGameSearch(), ullUniqueGameID, steamIDPlayer, EPlayerResult); } /// /// ends the game. no further SubmitPlayerResults for ullUniqueGameID will be accepted /// any future requests will provide a new ullUniqueGameID /// public static EGameSearchErrorCode_t EndGame(ulong ullUniqueGameID) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamGameSearch_EndGame(CSteamAPIContext.GetSteamGameSearch(), ullUniqueGameID); } } public static class SteamParties { /// /// ============================================================================================= /// Party Client APIs /// Enumerate any active beacons for parties you may wish to join /// public static uint GetNumActiveBeacons() { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamParties_GetNumActiveBeacons(CSteamAPIContext.GetSteamParties()); } public static PartyBeaconID_t GetBeaconByIndex(uint unIndex) { InteropHelp.TestIfAvailableClient(); return (PartyBeaconID_t)NativeMethods.ISteamParties_GetBeaconByIndex(CSteamAPIContext.GetSteamParties(), unIndex); } public static bool GetBeaconDetails(PartyBeaconID_t ulBeaconID, out CSteamID pSteamIDBeaconOwner, out SteamPartyBeaconLocation_t pLocation, out string pchMetadata, int cchMetadata) { InteropHelp.TestIfAvailableClient(); IntPtr pchMetadata2 = Marshal.AllocHGlobal(cchMetadata); bool ret = NativeMethods.ISteamParties_GetBeaconDetails(CSteamAPIContext.GetSteamParties(), ulBeaconID, out pSteamIDBeaconOwner, out pLocation, pchMetadata2, cchMetadata); pchMetadata = ret ? InteropHelp.PtrToStringUTF8(pchMetadata2) : null; Marshal.FreeHGlobal(pchMetadata2); return ret; } /// /// Join an open party. Steam will reserve one beacon slot for your SteamID, /// and return the necessary JoinGame string for you to use to connect /// public static SteamAPICall_t JoinParty(PartyBeaconID_t ulBeaconID) { InteropHelp.TestIfAvailableClient(); return (SteamAPICall_t)NativeMethods.ISteamParties_JoinParty(CSteamAPIContext.GetSteamParties(), ulBeaconID); } /// /// ============================================================================================= /// Party Host APIs /// Get a list of possible beacon locations /// public static bool GetNumAvailableBeaconLocations(out uint puNumLocations) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamParties_GetNumAvailableBeaconLocations(CSteamAPIContext.GetSteamParties(), out puNumLocations); } public static bool GetAvailableBeaconLocations(SteamPartyBeaconLocation_t[] pLocationList, uint uMaxNumLocations) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamParties_GetAvailableBeaconLocations(CSteamAPIContext.GetSteamParties(), pLocationList, uMaxNumLocations); } /// /// Create a new party beacon and activate it in the selected location. /// unOpenSlots is the maximum number of users that Steam will send to you. /// When people begin responding to your beacon, Steam will send you /// PartyReservationCallback_t callbacks to let you know who is on the way. /// public static SteamAPICall_t CreateBeacon(uint unOpenSlots, ref SteamPartyBeaconLocation_t pBeaconLocation, string pchConnectString, string pchMetadata) { InteropHelp.TestIfAvailableClient(); using (var pchConnectString2 = new InteropHelp.UTF8StringHandle(pchConnectString)) using (var pchMetadata2 = new InteropHelp.UTF8StringHandle(pchMetadata)) { return (SteamAPICall_t)NativeMethods.ISteamParties_CreateBeacon(CSteamAPIContext.GetSteamParties(), unOpenSlots, ref pBeaconLocation, pchConnectString2, pchMetadata2); } } /// /// Call this function when a user that had a reservation (see callback below) /// has successfully joined your party. /// Steam will manage the remaining open slots automatically. /// public static void OnReservationCompleted(PartyBeaconID_t ulBeacon, CSteamID steamIDUser) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamParties_OnReservationCompleted(CSteamAPIContext.GetSteamParties(), ulBeacon, steamIDUser); } /// /// To cancel a reservation (due to timeout or user input), call this. /// Steam will open a new reservation slot. /// Note: The user may already be in-flight to your game, so it's possible they will still connect and try to join your party. /// public static void CancelReservation(PartyBeaconID_t ulBeacon, CSteamID steamIDUser) { InteropHelp.TestIfAvailableClient(); NativeMethods.ISteamParties_CancelReservation(CSteamAPIContext.GetSteamParties(), ulBeacon, steamIDUser); } /// /// Change the number of open beacon reservation slots. /// Call this if, for example, someone without a reservation joins your party (eg a friend, or via your own matchmaking system). /// public static SteamAPICall_t ChangeNumOpenSlots(PartyBeaconID_t ulBeacon, uint unOpenSlots) { InteropHelp.TestIfAvailableClient(); return (SteamAPICall_t)NativeMethods.ISteamParties_ChangeNumOpenSlots(CSteamAPIContext.GetSteamParties(), ulBeacon, unOpenSlots); } /// /// Turn off the beacon. /// public static bool DestroyBeacon(PartyBeaconID_t ulBeacon) { InteropHelp.TestIfAvailableClient(); return NativeMethods.ISteamParties_DestroyBeacon(CSteamAPIContext.GetSteamParties(), ulBeacon); } /// /// Utils /// public static bool GetBeaconLocationData(SteamPartyBeaconLocation_t BeaconLocation, ESteamPartyBeaconLocationData eData, out string pchDataStringOut, int cchDataStringOut) { InteropHelp.TestIfAvailableClient(); IntPtr pchDataStringOut2 = Marshal.AllocHGlobal(cchDataStringOut); bool ret = NativeMethods.ISteamParties_GetBeaconLocationData(CSteamAPIContext.GetSteamParties(), BeaconLocation, eData, pchDataStringOut2, cchDataStringOut); pchDataStringOut = ret ? InteropHelp.PtrToStringUTF8(pchDataStringOut2) : null; Marshal.FreeHGlobal(pchDataStringOut2); return ret; } } } #endif // !DISABLESTEAMWORKS