From 10644244ba9da40766c247161ad064294e1e767a Mon Sep 17 00:00:00 2001 From: Eamonn Rea Date: Fri, 7 Jun 2024 18:45:25 +0100 Subject: [PATCH 1/5] Non-Steam Games: Add 'nsg' filter to 'list' command Allows for listing all Non-Steam Games with Name, AppID, Path, or all three. --- steamtinkerlaunch | 127 ++++++++++++++++++++++++++++------------------ 1 file changed, 77 insertions(+), 50 deletions(-) diff --git a/steamtinkerlaunch b/steamtinkerlaunch index adaac4aa..b27de9f5 100755 --- a/steamtinkerlaunch +++ b/steamtinkerlaunch @@ -7,7 +7,7 @@ PREFIX="/usr" PROGNAME="SteamTinkerLaunch" NICEPROGNAME="Steam Tinker Launch" -PROGVERS="v14.0.20240608-1" +PROGVERS="v14.0.20240608-2 (list-all-nonsteam-games)" PROGCMD="${0##*/}" PROGINTERNALPROTNAME="Proton-stl" SHOSTL="stl" @@ -7781,78 +7781,96 @@ function pickGameWindowNameMeta { # Can take two types of commands: # - `steamtinkerlaunch list owned/installed` - Returns "Game Name (AppID)" # - `steamtinkerlaunch list owned/installed id/name/path/full` - Returns either AppID, Game Name, Game Paths, or all in the format "Game Name (AppID) -> /path/to/game" +# +# This function is not very efficient, Non-Steam Games in particular are inefficient because we read shortcuts.vdf each time we want to parse info +# We parse it once to get the IDs, then again in each `getTitleFromID` and `getGameDir` call. This makes it pretty slow +# It works for now, but in future we should enhance it function listSteamGames { function getGameCount { + TOTALGAMESOWNEDPRINTFSTR="" + if [ "$LSFILTER" == "owned" ] || [ "$LSFILTER" == "o" ]; then - printf "Total games owned: %s\n" "${#LISTAIDSARR[@]}" + TOTALGAMESOWNEDPRINTFSTR="Total games owned" + elif [ "$LSFILTER" == "non-steam" ] || [ "$LSFILTER" == "nsg" ]; then + TOTALGAMESOWNEDPRINTFSTR="Total Non-Steam Games in library" else - printf "Total games installed: %s\n" "${#LISTAIDSARR[@]}" + TOTALGAMESOWNEDPRINTFSTR="Total games installed" fi + + printf "${TOTALGAMESOWNEDPRINTFSTR}: %s\n" "${#LISTAIDSARR[@]}" } - LSFILTER="$1" # e.g. "owned", "installed" + LSFILTER="$1" # e.g. "owned", "installed", "non-steam" LSTYPE="$2" # e.g. "id", "name", "path", "count", "full" LISTAIDS="" + SEARCHSTEAMSHORTCUTS="0" + if [ "$LSFILTER" == "owned" ] || [ "$LSFILTER" == "o" ]; then LISTAIDS="$( getOwnedAids )" elif [ "$LSFILTER" == "installed" ] || [ "$LSFILTER" == "i" ]; then - if [ "$(listInstalledGameIDs | wc -l)" -eq 0 ]; then - writelog "SKIP" "${FUNCNAME[0]} - No installed games found!" "E" - echo "No installed games found!" - - exit - else - LISTAIDS="$( listInstalledGameIDs )" - fi + LISTAIDS="$( listInstalledGameIDs )" + elif [ "$LSFILTER" == "non-steam" ] || [ "$LSFILTER" == "nsg" ]; then + LISTAIDS="$( listNonSteamGameIDs )" + SEARCHSTEAMSHORTCUTS="1" # Only search Steam Shortcuts if we passed that filter type else + writelog "INFO" "${FUNCNAME[0]} - Unknown argument passed to 'list' command - '$LSFILTER'" echo "unknown argument passed to 'list' command - '$LSFILTER'" + + exit fi - if [ -n "$LISTAIDS" ]; then - readarray -t LISTAIDSARR <<<"$LISTAIDS" + if [ -z "$LISTAIDS" ]; then + writelog "SKIP" "${FUNCNAME[0]} - No games found for given filter '$LSFILTER'" + echo "No games found for filter '$LSFILTER'." - if [ "$LSTYPE" == "id" ]; then - for AID in "${LISTAIDSARR[@]}"; do - echo "$AID" - done - elif [ "$LSTYPE" == "name" ]; then + exit + fi + + readarray -t LISTAIDSARR <<<"$LISTAIDS" + + if [ "$LSTYPE" == "id" ]; then + for AID in "${LISTAIDSARR[@]}"; do + echo "$AID" + done + elif [ "$LSTYPE" == "name" ]; then + for AID in "${LISTAIDSARR[@]}"; do + getTitleFromID "${AID}" "${SEARCHSTEAMSHORTCUTS}" + done + elif [ "$LSTYPE" == "path" ]; then + if [ "$LSFILTER" == "owned" ] || [ "$LSFILTER" == "o" ]; then + echo "Cannot use 'path' option when returning 'owned' games, as not all owned games will have an installation path!" + echo "Use another option instead, or leave blank to only return path for games which have an installation path." + + exit + else for AID in "${LISTAIDSARR[@]}"; do - getTitleFromID "${AID}" + getGameDir "$AID" "1" "${SEARCHSTEAMSHORTCUTS}" done - elif [ "$LSTYPE" == "path" ]; then - if [ "$LSFILTER" == "owned" ] || [ "$LSFILTER" == "o" ]; then - echo "Cannot use 'path' option when returning 'owned' games, as not all owned games will have an installation path!" - echo "Use another option instead, or leave blank to only return path for games which have an installation path." - else - for AID in "${LISTAIDSARR[@]}"; do - getGameDir "$AID" "1" - done - fi - elif [ "$LSTYPE" == "count" ]; then - printf "\n%s" "$( getGameCount )" - elif [ "$LSTYPE" == "full" ] || [ -z "$LSTYPE" ]; then # This is the default if id/name/path/full is not passed - for AID in "${LISTAIDSARR[@]}"; do - GAMDIR="$( getGameDir "$AID" )" - GAMDIREXISTS=$? - - # Only display game dir if the game is installed, i.e. if getGameDir does not return 1 - # This means we won't return an error if we're returning OWNED games, as some owned games may not have paths - if [ "$GAMDIREXISTS" -eq 1 ]; then - GAMNAM="$( getTitleFromID "$AID" )" - GAMNAMEXISTS=$? - if [ "$GAMNAMEXISTS" -eq 1 ]; then - echo "$AID" # Game name unknown, probably never installed before? Just return AppID in this case - else - echo "$GAMNAM ($AID)" - fi + fi + elif [ "$LSTYPE" == "count" ]; then + printf "\n%s" "$( getGameCount )" + elif [ "$LSTYPE" == "full" ] || [ -z "$LSTYPE" ]; then # This is the default if id/name/path/full is not passed + for AID in "${LISTAIDSARR[@]}"; do + GAMDIR="$( getGameDir "$AID" "" "${SEARCHSTEAMSHORTCUTS}" )" + GAMDIREXISTS=$? + + # Only display game dir if the game is installed, i.e. if getGameDir does not return 1 + # This means we won't return an error if we're returning OWNED games, as some owned games may not have paths + if [ "$GAMDIREXISTS" -eq 1 ]; then + GAMNAM="$( getTitleFromID "$AID" "${SEARCHSTEAMSHORTCUTS}" )" + GAMNAMEXISTS=$? + if [ "$GAMNAMEXISTS" -eq 1 ]; then + echo "$AID" # Game name unknown, probably never installed before? Just return AppID in this case else - echo "$GAMDIR" + echo "$GAMNAM ($AID)" fi - done + else + echo "$GAMDIR" + fi + done - printf "\n%s" "$( getGameCount )" # Show total for "full" - fi + printf "\n%s\n" "$( getGameCount )" # Show total for "full" fi } @@ -8320,6 +8338,15 @@ function getSteamShortcutsVdfFileHex { xxd -p -c 0 "$SCPATH" } +function listNonSteamGameIDs { + NONSTEAMGAMEAIDS=() + + writelog "INFO" "${FUNCNAME[0]} - Reading all Non-Steam AppIDs from shortcuts.vdf" + while read -r SCVDFE; do + parseSteamShortcutEntryAppID "$SCVDFE" + done <<< "$( getSteamShortcutHex )" +} + function haveAnySteamShortcuts { if [ "$( getSteamShortcutHex | wc -c )" -gt 0 ]; then return 0 From 515048c097b3e0da77954ddde4fc8d1d1cf13e29 Mon Sep 17 00:00:00 2001 From: Eamonn Rea Date: Fri, 7 Jun 2024 18:47:16 +0100 Subject: [PATCH 2/5] update help screen --- steamtinkerlaunch | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/steamtinkerlaunch b/steamtinkerlaunch index b27de9f5..2d3f35e5 100755 --- a/steamtinkerlaunch +++ b/steamtinkerlaunch @@ -7779,8 +7779,8 @@ function pickGameWindowNameMeta { # General function for the "steamtinkerlaunch list function" # Can take two types of commands: -# - `steamtinkerlaunch list owned/installed` - Returns "Game Name (AppID)" -# - `steamtinkerlaunch list owned/installed id/name/path/full` - Returns either AppID, Game Name, Game Paths, or all in the format "Game Name (AppID) -> /path/to/game" +# - `steamtinkerlaunch list owned/installed/non-steam` - Returns "Game Name (AppID)" +# - `steamtinkerlaunch list owned/installed/non-steam id/name/path/full` - Returns either AppID, Game Name, Game Paths, or all in the format "Game Name (AppID) -> /path/to/game" # # This function is not very efficient, Non-Steam Games in particular are inefficient because we read shortcuts.vdf each time we want to parse info # We parse it once to get the IDs, then again in each `getTitleFromID` and `getGameDir` call. This makes it pretty slow @@ -7849,7 +7849,7 @@ function listSteamGames { done fi elif [ "$LSTYPE" == "count" ]; then - printf "\n%s" "$( getGameCount )" + printf "\n%s\n" "$( getGameCount )" elif [ "$LSTYPE" == "full" ] || [ -z "$LSTYPE" ]; then # This is the default if id/name/path/full is not passed for AID in "${LISTAIDSARR[@]}"; do GAMDIR="$( getGameDir "$AID" "" "${SEARCHSTEAMSHORTCUTS}" )" @@ -22194,7 +22194,7 @@ function howto { echo " auto Create/Download data for all installed games first" echo " update ReCreate all Collection Menus" echo " Can be combined with auto" - echo " list List ids of games" + echo " list List info on games" echo " Optionally specify whether you want to see" echo " each game's " echo " listproton|lp List name and path of all Proton versions known to SteamTinkerLaunch" From 876e5d7b8752dadbf7fd4ed54928590d8e00425a Mon Sep 17 00:00:00 2001 From: Eamonn Rea Date: Fri, 7 Jun 2024 18:59:15 +0100 Subject: [PATCH 3/5] formatting --- steamtinkerlaunch | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/steamtinkerlaunch b/steamtinkerlaunch index 2d3f35e5..62aa13f0 100755 --- a/steamtinkerlaunch +++ b/steamtinkerlaunch @@ -7785,6 +7785,9 @@ function pickGameWindowNameMeta { # This function is not very efficient, Non-Steam Games in particular are inefficient because we read shortcuts.vdf each time we want to parse info # We parse it once to get the IDs, then again in each `getTitleFromID` and `getGameDir` call. This makes it pretty slow # It works for now, but in future we should enhance it +# +# One potential way to enhance this function is to split it out into a separate function for each filter type, but we would need +# to consider how this function is used by other parts of the codebase and if that could be disruptive. function listSteamGames { function getGameCount { TOTALGAMESOWNEDPRINTFSTR="" @@ -22187,7 +22190,7 @@ function howto { echo " Note that this will not remove your mods or installed Winetricks" echo " lang=