Skip to content

Commit 4c9808b

Browse files
authored
[UX] Improve Compatibility layer setup and handling (#761)
* Add compatibility warning for non-native games * feat: download gptk even if wine is available * fix: winetricks download on macos * feat: set gptk as default on mac * feat: style the warning on install dialog * fix: lint * feat: show warning dialog when running a game with compatibility layer * feat: add method to check if mac is GPTK compatible * feat: show a warning if system is not compatible with GPTK * fix: lint * fix: check for gptk on validWine as well * fix: show correct dialog when wine not found * feat: Add isRosettaAvailable function to check for Rosetta availability on macOS * fix: PR Comments * fix: pr comments * fix: lint --------- Co-authored-by: Flavio F Lima <flavioislima@users.noreply.github.com>
1 parent d42a4bb commit 4c9808b

File tree

18 files changed

+1401
-1137
lines changed

18 files changed

+1401
-1137
lines changed

public/locales/en/gamepage.json

+224-218
Large diffs are not rendered by default.

public/locales/en/login.json

+18-18
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,18 @@
1-
{
2-
"button": {
3-
"error": "Error, try a different Code",
4-
"loading": "Loading",
5-
"login": "Log in"
6-
},
7-
"message": {
8-
"part1": "In order to log in and install your games, you first need to follow the steps below:",
9-
"part2": "Open the",
10-
"part3": "Epic Store here",
11-
"part4": ", log in to your account and copy your",
12-
"part5": "authorization code information number",
13-
"part6": "Paste your",
14-
"part7": "authorization code number",
15-
"part8": "in the input box below and click on the login button."
16-
},
17-
"welcome": "Welcome!"
18-
}
1+
{
2+
"button": {
3+
"error": "Error, try a different Code",
4+
"loading": "Loading",
5+
"login": "Log in"
6+
},
7+
"message": {
8+
"part1": "In order to log in and install your games, you first need to follow the steps below:",
9+
"part2": "Open the",
10+
"part3": "Epic Store here",
11+
"part4": ", log in to your account and copy your",
12+
"part5": "authorization code information number",
13+
"part6": "Paste your",
14+
"part7": "authorization code number",
15+
"part8": "in the input box below and click on the login button."
16+
},
17+
"welcome": "Welcome!"
18+
}

public/locales/en/translation.json

+874-874
Large diffs are not rendered by default.

src/backend/launcher.ts

+10-2
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,8 @@ import {
2424
searchForExecutableOnPath,
2525
quoteIfNecessary,
2626
errorHandler,
27-
removeQuoteIfNecessary
27+
removeQuoteIfNecessary,
28+
isMacSonomaOrHigher
2829
} from './utils'
2930
import {
3031
logDebug,
@@ -502,7 +503,7 @@ function setupWrappers(
502503
* @returns true if the wine version exists, false if it doesn't
503504
*/
504505
export async function validWine(
505-
wineVersion: WineInstallation
506+
wineVersion: WineInstallation | undefined
506507
): Promise<boolean> {
507508
if (!wineVersion) {
508509
return false
@@ -518,6 +519,13 @@ export async function validWine(
518519
const necessary = type === 'wine' ? [bin, wineserver] : [bin]
519520
const haveAll = necessary.every((binary) => existsSync(binary as string))
520521

522+
if (isMac && type === 'toolkit') {
523+
const isGPTKCompatible = await isMacSonomaOrHigher()
524+
if (!isGPTKCompatible) {
525+
return false
526+
}
527+
}
528+
521529
// if wine version does not exist, use the default one
522530
if (!haveAll) {
523531
return false

src/backend/main.ts

+13-2
Original file line numberDiff line numberDiff line change
@@ -66,8 +66,10 @@ import {
6666
getSystemInfo,
6767
handleExit,
6868
isEpicServiceOffline,
69+
checkRosettaInstall,
6970
openUrlOrFile,
7071
resetApp,
72+
setGPTKDefaultOnMacOS,
7173
showAboutWindow,
7274
showItemInFolder,
7375
wait
@@ -90,6 +92,7 @@ import {
9092
isCLIFullscreen,
9193
isCLINoGui,
9294
isFlatpak,
95+
isMac,
9396
isSteamDeckGameMode,
9497
onboardLocalStore,
9598
publicDir,
@@ -296,11 +299,19 @@ async function initializeWindow(): Promise<BrowserWindow> {
296299

297300
setTimeout(async () => {
298301
// Will download Wine if none was found
299-
const availableWine = await GlobalConfig.get().getAlternativeWine()
302+
const availableWine = (await GlobalConfig.get().getAlternativeWine()) || []
303+
const toolkitListDownloaded = availableWine.some(
304+
(wine) => wine.type === 'toolkit'
305+
)
306+
const shouldDownloadWine =
307+
!availableWine.length || (isMac && !toolkitListDownloaded)
308+
300309
Promise.all([
301310
DXVK.getLatest(),
302311
Winetricks.download(),
303-
!availableWine.length ? downloadDefaultWine() : null
312+
shouldDownloadWine ? downloadDefaultWine() : null,
313+
isMac && checkRosettaInstall(),
314+
isMac && setGPTKDefaultOnMacOS()
304315
])
305316
}, 2500)
306317

src/backend/tools.ts

+3-6
Original file line numberDiff line numberDiff line change
@@ -393,15 +393,12 @@ export const DXVK = {
393393

394394
export const Winetricks = {
395395
download: async () => {
396-
if (!isLinux) {
396+
if (isWindows) {
397397
return
398398
}
399399

400-
const linuxUrl =
400+
const url =
401401
'https://raw.githubusercontent.com/Winetricks/winetricks/master/src/winetricks'
402-
const macUrl =
403-
'https://raw.githubusercontent.com/The-Wineskin-Project/winetricks/macOS/src/winetricks'
404-
const url = isMac ? macUrl : linuxUrl
405402
const path = `${toolsPath}/winetricks`
406403

407404
if (!isOnline()) {
@@ -432,7 +429,7 @@ export const Winetricks = {
432429
}
433430

434431
return new Promise<void>((resolve) => {
435-
const winetricks = `${toolsPath}/winetricks`
432+
const winetricks = join(toolsPath, 'winetricks')
436433

437434
const { winePrefix, wineBin } = getWineFromProton(
438435
wineVersion,

src/backend/utils.ts

+128-4
Original file line numberDiff line numberDiff line change
@@ -899,6 +899,27 @@ async function ContinueWithFoundWine(
899899
selectedWine: string,
900900
foundWine: string
901901
): Promise<{ response: number }> {
902+
const isGPTK = selectedWine.toLowerCase().includes('toolkit')
903+
const isGPTKCompatible = await isMacSonomaOrHigher()
904+
905+
if (isMac && isGPTK && !isGPTKCompatible) {
906+
const { response } = await dialog.showMessageBox({
907+
title: i18next.t(
908+
'box.warning.wine-change.title-gptk',
909+
'Game Porting Toolkit Not Compatible '
910+
),
911+
message: i18next.t('box.warning.wine-change.message-gptk', {
912+
defaultValue:
913+
'To be able to run games using the Apple Gaming porting toolkit you need to upgrade your macOS to 14 (Sonoma) or higher. {{newline}} We found Wine on your system, do you want to continue launching using {{foundWine}} ?',
914+
newline: '\n',
915+
foundWine: foundWine
916+
}),
917+
buttons: [i18next.t('box.yes'), i18next.t('box.no')],
918+
icon: icon
919+
})
920+
return { response }
921+
}
922+
902923
const { response } = await dialog.showMessageBox({
903924
title: i18next.t('box.warning.wine-change.title', 'Wine not found!'),
904925
message: i18next.t('box.warning.wine-change.message', {
@@ -915,19 +936,79 @@ async function ContinueWithFoundWine(
915936
return { response }
916937
}
917938

939+
export async function checkRosettaInstall() {
940+
if (!isMac) {
941+
return
942+
}
943+
944+
// check if on arm64 macOS
945+
const { stdout: archCheck } = await execAsync('arch')
946+
const isArm64 = archCheck.trim() === 'arm64'
947+
948+
if (!isArm64) {
949+
return
950+
}
951+
952+
const { stdout: rosettaCheck } = await execAsync(
953+
'arch -x86_64 /usr/sbin/sysctl sysctl.proc_translated'
954+
)
955+
956+
const result = rosettaCheck.split(':')[1].trim() === '1'
957+
958+
logInfo(
959+
`Rosetta is ${result ? 'available' : 'not available'} on this system.`,
960+
LogPrefix.Backend
961+
)
962+
963+
if (!result) {
964+
// show a dialog saying that hyperplay wont run without rosetta and add information on how to install it
965+
await dialog.showMessageBox({
966+
title: i18next.t('box.warning.rosetta.title', 'Rosetta not found'),
967+
message: i18next.t(
968+
'box.warning.rosetta.message',
969+
'HyperPlay requires Rosetta to run correctly on macOS with Apple Silicon chips. Please install it from the macOS terminal using the following command: "softwareupdate --install-rosetta" and restart HyperPlay. '
970+
),
971+
buttons: ['OK'],
972+
icon: icon
973+
})
974+
975+
logInfo(
976+
'Rosetta is not available, install it with softwareupdate --install-rosetta from the terminal',
977+
LogPrefix.Backend
978+
)
979+
}
980+
}
981+
982+
export async function isMacSonomaOrHigher() {
983+
if (!isMac) {
984+
return false
985+
}
986+
logInfo('Checking if macOS is Sonoma or higher', LogPrefix.Backend)
987+
988+
const { release } = await si.osInfo()
989+
const [major] = release.split('.').map(Number)
990+
991+
return major >= 14
992+
}
993+
918994
export async function downloadDefaultWine() {
919995
// refresh wine list
920996
await updateWineVersionInfos(true)
921997
// get list of wines on wineDownloaderInfoStore
922998
const availableWine = wineDownloaderInfoStore.get('wine-releases', [])
923999
// use Wine-GE type if on Linux and Wine-Crossover if on Mac
924-
const release = availableWine.filter((version) => {
1000+
const release = availableWine.filter(async (version) => {
9251001
if (isLinux) {
9261002
return (
9271003
version.type === 'Wine-GE' && version.version.includes('Wine-GE-Proton')
9281004
)
9291005
} else if (isMac) {
930-
return version.type === 'Game-Porting-Toolkit'
1006+
const isGPTKCompatible = await isMacSonomaOrHigher()
1007+
1008+
if (isGPTKCompatible) {
1009+
return version.type === 'Game-Porting-Toolkit'
1010+
}
1011+
return version.type === 'Wine-Crossover'
9311012
}
9321013
return false
9331014
})[0]
@@ -969,14 +1050,46 @@ export async function downloadDefaultWine() {
9691050
return null
9701051
}
9711052

1053+
export async function setGPTKDefaultOnMacOS() {
1054+
const isGPTKCompatible = await isMacSonomaOrHigher()
1055+
if (!isGPTKCompatible) {
1056+
return
1057+
}
1058+
1059+
const wineList = await GlobalConfig.get().getAlternativeWine()
1060+
const gptk = wineList.find((wine) => wine.type === 'toolkit')
1061+
if (!gptk) {
1062+
await downloadDefaultWine()
1063+
return setGPTKDefaultOnMacOS()
1064+
}
1065+
if (gptk && existsSync(gptk.bin)) {
1066+
logInfo(`Changing wine version to ${gptk.name}`)
1067+
GlobalConfig.get().setSetting('wineVersion', gptk)
1068+
// update prefix to use the new one as well
1069+
const installPath = GlobalConfig.get().getSettings().defaultInstallPath
1070+
const newPrefix = join(installPath, 'Prefixes', 'GPTK')
1071+
GlobalConfig.get().setSetting('winePrefix', newPrefix)
1072+
}
1073+
return
1074+
}
1075+
9721076
export async function checkWineBeforeLaunch(
9731077
appName: string,
9741078
gameSettings: GameSettings,
9751079
logFileLocation: string
9761080
): Promise<boolean> {
9771081
const wineIsValid = await validWine(gameSettings.wineVersion)
9781082

979-
if (wineIsValid) {
1083+
const isToolkit = gameSettings.wineVersion.type === 'toolkit'
1084+
const isGPTKCompatible = await isMacSonomaOrHigher()
1085+
1086+
const isValidOnLinux = isLinux && wineIsValid
1087+
const isValidtoolkitOnMac =
1088+
isMac && isToolkit && isGPTKCompatible && wineIsValid
1089+
const isValidWineOnMac = isMac && !isToolkit && wineIsValid
1090+
const isValidOnMac = isValidtoolkitOnMac || isValidWineOnMac
1091+
1092+
if (isValidOnMac || isValidOnLinux) {
9801093
return true
9811094
} else {
9821095
if (!logsDisabled) {
@@ -1011,7 +1124,18 @@ export async function checkWineBeforeLaunch(
10111124
}
10121125
} else {
10131126
const wineList = await GlobalConfig.get().getAlternativeWine()
1014-
const firstFoundWine = wineList[0]
1127+
1128+
// if Linux get the first element, if macOS and isGPTKCompatible is true get one with type 'toolkit', otherwise get the one with type 'wine'
1129+
const firstFoundWine = wineList.find((wine) => {
1130+
if (isLinux) {
1131+
return wine.type === 'wine'
1132+
} else if (isMac) {
1133+
return isGPTKCompatible
1134+
? wine.type === 'toolkit'
1135+
: wine.type === 'wine'
1136+
}
1137+
return undefined
1138+
})
10151139

10161140
const isValidWine = await validWine(firstFoundWine)
10171141

src/frontend/components/UI/DialogHandler/components/MessageBoxModal/index.tsx

+11
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,9 @@ interface MessageBoxModalProps {
1717
onClose: () => void
1818
buttons: Array<ButtonOptions>
1919
type: DialogType
20+
showCheckbox?: boolean
21+
checkboxLabel?: string
22+
checkboxValue?: boolean
2023
}
2124

2225
const MessageBoxModal: React.FC<MessageBoxModalProps> = function (props) {
@@ -72,7 +75,15 @@ const MessageBoxModal: React.FC<MessageBoxModalProps> = function (props) {
7275
<DialogHeader onClose={props.onClose}>{props.title}</DialogHeader>
7376
<DialogContent className="body dialogContent">
7477
{getContent()}
78+
{props.showCheckbox && (
79+
<div className="checkbox">
80+
<input type="checkbox" checked={props.checkboxValue ? true : false}>
81+
{props.checkboxLabel}
82+
</input>
83+
</div>
84+
)}
7585
</DialogContent>
86+
7687
<DialogFooter>{getButtons()}</DialogFooter>
7788
</Dialog>
7889
)

src/frontend/components/UI/DialogHandler/index.tsx

+10-1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,15 @@ export default function DialogHandler() {
2525
}
2626
}, [])
2727

28+
const handleClose = () => {
29+
if (dialogModalOptions.onClose) {
30+
// runs the optional onClose function and then closes the modal
31+
dialogModalOptions.onClose()
32+
return showDialogModal({ showDialog: false })
33+
}
34+
showDialogModal({ showDialog: false })
35+
}
36+
2837
return (
2938
<>
3039
{dialogModalOptions.showDialog && (
@@ -33,7 +42,7 @@ export default function DialogHandler() {
3342
title={dialogModalOptions.title ? dialogModalOptions.title : ''}
3443
message={dialogModalOptions.message ? dialogModalOptions.message : ''}
3544
buttons={dialogModalOptions.buttons ? dialogModalOptions.buttons : []}
36-
onClose={() => showDialogModal({ showDialog: false })}
45+
onClose={() => handleClose()}
3746
/>
3847
)}
3948
</>

0 commit comments

Comments
 (0)