2 * Copyright (C) 2021-2023 Mt.ZERO Software, Harry Godden - All Rights Reserved
3 * All trademarks are property of their respective owners
10 #include "vg/vg_steam.h"
11 #include "vg/vg_steam_utils.h"
12 #include "vg/vg_steam_networking.h"
13 #include "vg/vg_steam_auth.h"
14 #include "vg/vg_steam_http.h"
15 #include "vg/vg_steam_friends.h"
16 #include "vg/vg_steam_user_stats.h"
17 #include "submodules/anyascii/impl/c/anyascii.c"
20 * We only want to use steamworks if building for the networked version,
21 * theres not much point otherwise. We mainly want steamworks for setting
22 * achievements etc.. so that includes our own server too.
24 * This file also wraps the functions and interfaces that we want to use to
25 * make them a bit easier to read, since they are the flat API they have very
26 * long names. in non-networked builds they will return default errors or do
30 static char steam_username_at_startup
[128] = "Unassigned";
32 static void recv_steam_warning( int severity
, const char *msg
)
35 vg_low( "%s\n", msg
);
37 vg_info( "%s\n", msg
);
40 static int steam_ready
= 0,
41 steam_stats_ready
= 0;
43 static void *hSteamNetworkingSockets
,
46 static ISteamUserStats
*hSteamUserStats
;
47 static HSteamPipe hSteamClientPipe
;
49 static const char *steam_achievement_names
[] =
51 "ALBERT", "MARC", "JANET", "BERNADETTA",
52 "ROUTE_MPY", "ROUTE_MPG", "ROUTE_MPB", "ROUTE_MPR",
53 "ROUTE_TO", "ROUTE_TC", "CITY_COMPLETE", "MTZERO_SILVER", "MTZERO_GOLD",
57 static void steam_store_achievements(void)
59 if( steam_ready
&& steam_stats_ready
){
60 SteamAPI_ISteamUserStats_StoreStats( hSteamUserStats
);
64 static void update_ach_models(void);
65 static void steam_set_achievement( const char *name
){
67 if( !strcmp(name
,"MARC") ) skaterift
.achievements
|= 0x1;
68 if( !strcmp(name
,"ALBERT") ) skaterift
.achievements
|= 0x2;
69 if( !strcmp(name
,"JANET") ) skaterift
.achievements
|= 0x4;
70 if( !strcmp(name
,"BERNADETTA") ) skaterift
.achievements
|= 0x8;
73 if( steam_ready
&& steam_stats_ready
){
74 if( SteamAPI_ISteamUserStats_SetAchievement( hSteamUserStats
, name
) ){
75 vg_success( "Achievement set! '%s'\n", name
);
79 vg_warn( "Failed to set achievement: %s\n", name
);
83 vg_warn( "Failed to set achievement (steam not ready): %s\n", name
);
87 static void steam_clear_achievement( const char *name
)
89 if( steam_ready
&& steam_stats_ready
){
90 if( SteamAPI_ISteamUserStats_ClearAchievement( hSteamUserStats
, name
) ){
91 vg_info( "Achievement cleared: '%s'\n", name
);
94 vg_warn( "Failed to clear achievement: %s\n", name
);
98 vg_warn( "Failed to clear achievement (steam not ready): %s\n", name
);
103 static void steam_print_all_achievements(void){
104 vg_info( "Achievements: \n" );
106 if( steam_ready
&& steam_stats_ready
){
107 for( int i
=0; i
<vg_list_size(steam_achievement_names
); i
++ ){
108 steamapi_bool set
= 0;
109 const char *name
= steam_achievement_names
[i
];
111 if( SteamAPI_ISteamUserStats_GetAchievement(
112 hSteamUserStats
, name
, &set
) )
114 vg_info( " %s %s\n", (set
? "[YES]": "[ ]"), name
);
117 vg_warn( " Error while fetching achievement status '%s'\n", name
);
122 vg_warn( " Steam is not initialized, no results\n" );
126 static int steam_achievement_ccmd( int argc
, char const *argv
[] )
128 if( !(steam_ready
&& steam_stats_ready
) ) return 1;
131 if( !strcmp( argv
[0], "list" ) ){
132 steam_print_all_achievements();
135 else if( !strcmp( argv
[0], "clearall" )){
136 for( int i
=0; i
<vg_list_size(steam_achievement_names
); i
++ )
137 steam_clear_achievement( steam_achievement_names
[i
] );
139 steam_store_achievements();
144 if( !strcmp( argv
[0], "set" ) ){
145 steam_set_achievement( argv
[1] );
146 steam_store_achievements();
149 else if( strcmp( argv
[0], "clear" ) ){
150 steam_clear_achievement( argv
[1] );
151 steam_store_achievements();
159 static void steam_on_recieve_current_stats( CallbackMsg_t
*msg
)
161 UserStatsReceived_t
*rec
= (UserStatsReceived_t
*)msg
->m_pubParam
;
163 if( rec
->m_eResult
== k_EResultOK
){
164 vg_info( "Recieved stats for: %lu (user: %lu)\n", rec
->m_nGameID
,
165 rec
->m_steamIDUser
);
166 steam_stats_ready
= 1;
168 steamapi_bool set
= 0;
169 if( SteamAPI_ISteamUserStats_GetAchievement(
170 hSteamUserStats
, "MARC", &set
) ){
171 if( set
) skaterift
.achievements
|= 0x1;
173 if( SteamAPI_ISteamUserStats_GetAchievement(
174 hSteamUserStats
, "ALBERT", &set
) ){
175 if( set
) skaterift
.achievements
|= 0x2;
177 if( SteamAPI_ISteamUserStats_GetAchievement(
178 hSteamUserStats
, "JANET", &set
) ){
179 if( set
) skaterift
.achievements
|= 0x4;
181 if( SteamAPI_ISteamUserStats_GetAchievement(
182 hSteamUserStats
, "BERNADETTA", &set
) ){
183 if( set
) skaterift
.achievements
|= 0x8;
188 vg_error( "Error recieveing stats for user (%u)\n", rec
->m_eResult
);
192 static u32
utf8_byte0_byte_count( u8 char0
)
194 for( u32 k
=2; k
<4; k
++ ){
195 if( !(char0
& (0x80 >> k
)) )
202 static u32
str_utf8_collapse( const char *str
, char *buf
, u32 length
){
203 u8
*ustr
= (u8
*)str
;
204 u32 utf32_code
= 0x00000000;
205 u32 i
=0, j
=0, utf32_byte_ct
=0;
208 if( ustr
[i
] == 0x00 )
211 if( ustr
[i
] & 0x80 ){
214 utf32_code
|= (ustr
[i
] & 0x3F) << (utf32_byte_ct
*6);
216 if( !utf32_byte_ct
){
218 size_t chars
= anyascii( utf32_code
, &match
);
220 for( u32 k
=0; k
<VG_MIN(chars
, length
-1-j
); k
++ ){
221 buf
[ j
++ ] = (u8
)match
[k
];
226 utf32_byte_ct
= utf8_byte0_byte_count( ustr
[i
] )-1;
227 utf32_code
= ustr
[i
] & (0x3F >> utf32_byte_ct
);
228 utf32_code
<<= utf32_byte_ct
*6;
232 utf32_byte_ct
= 0x00;
243 static int steam_init(void){
244 const char *username
= "offline player";
247 vg_info( "Initializing steamworks\n" );
249 if( !SteamAPI_Init() ){
251 vg_error( "Steamworks failed to initialize\n" );
257 SteamAPI_ManualDispatch_Init();
259 /* Connect interfaces */
260 hSteamClientPipe
= SteamAPI_GetHSteamPipe();
261 hSteamNetworkingSockets
= SteamAPI_SteamNetworkingSockets_SteamAPI();
262 hSteamUser
= SteamAPI_SteamUser();
264 ISteamUtils
*utils
= SteamAPI_SteamUtils();
265 SteamAPI_ISteamUtils_SetWarningMessageHook( utils
, recv_steam_warning
);
268 vg_success( "\nSteamworks API running\n" );
270 ISteamFriends
*hSteamFriends
= SteamAPI_SteamFriends();
271 username
= SteamAPI_ISteamFriends_GetPersonaName( hSteamFriends
);
275 * --------------------------------------------------------
277 hSteamUserStats
= SteamAPI_SteamUserStats();
278 steam_register_callback( k_iUserStatsReceived
,
279 steam_on_recieve_current_stats
);
281 if( !SteamAPI_ISteamUserStats_RequestCurrentStats( hSteamUserStats
) )
282 vg_warn( "No Steam Logon: Cannot request stats\n" );
285 vg_console_reg_cmd( "ach", steam_achievement_ccmd
, NULL
);
289 /* TODO: On username update callback */
290 str_utf8_collapse( username
, steam_username_at_startup
,
291 vg_list_size(steam_username_at_startup
) );
296 static void steam_update(void)
299 steamworks_event_loop( hSteamClientPipe
);
303 static void steam_end(void)
306 vg_info( "Shutting down\n..." );