wine/tools/winelauncher.in
2001-01-17 01:47:18 +00:00

521 lines
18 KiB
Bash
Executable File

#!/bin/sh
#------------------------------------------------------------------------------
# Winelauncher
# This shell script attempts to intelligently manage the process
# of launching a program with Wine. It adds some level of
# visual feedback to an end user.
#
# Usage:
# winelauncher [options] "<windows program> [program arguments]"
#
# This script is meant to be installed to /usr/bin/wine, and
# to be used to invoke a Windows executable.
# The options are passed through directly to Wine, and are
# documented in the Wine man page.
#
# Copyright (c) 2000 by Jeremy White for CodeWeavers
#
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# Primary configuration area - change this if you installed Wine to
# a location other than @prefix@
#------------------------------------------------------------------------------
prefix=@prefix@
#------------------------------------------------------------------------------
# Secondary configuration area; change these at your own risk.
#------------------------------------------------------------------------------
exec_prefix=@exec_prefix@
WINEBIN=@bindir@
WINELIB=@libdir@
WINESERVERBIN=
WINELIBDLLS=
#------------------------------------------------------------------------------
# Establish Color Scheme
#------------------------------------------------------------------------------
COLOR=' -xrm *.Command.background:darkgrey
-xrm *.Command.foreground:black
-xrm *.Text.background:black
-xrm *.Text.foreground:green
-xrm *.Form.background:grey
-xrm *.Form.foreground:green
-xrm *.foreground:green
-xrm *.background:black'
#------------------------------------------------------------------------------
# Locate either xmessage or gmessage, if we can.
#------------------------------------------------------------------------------
type xmessage >/dev/null 2>/dev/null
if [ $? -ne 0 ] ; then
echo "
Warning:
The CodeWeavers Wine launcher is unable to find xmessage.
This launcher script relies heavily on finding this tool,
and without it, it will behave poorly.
Most Linux distributions have one or the other of these
tools.
We strongly recommend that you use your distributions
software methods to locate xmessage."
else
XMESSAGE="xmessage $COLOR"
fi
#------------------------------------------------------------------------------
# We're going to do a lot of fancy footwork below.
# Before we get started, it would be nice to know the argv0,
# of the actual script we're running (and lets remove at least
# one level of symlinking).
#------------------------------------------------------------------------------
real_name=`find $0 -type l -printf "%l\n"`
if [ ! $real_name ]; then
real_name=$0;
fi
argv0_dir=`find $real_name -printf "%h\n"`
if [ -z $argv0_dir ] ; then
argv0_dir=.
fi
#------------------------------------------------------------------------------
# Okay, now all that junk above was established at configure time.
# However, if this is an RPM install, they may have chosen
# to relocate this installation. If so, that stuff above
# is all broken and we should rejigger it.
#------------------------------------------------------------------------------
WINE_BIN_NAME=wine.bin
if [ ! -x $WINEBIN/$WINE_BIN_NAME ] ; then
WINEBIN=`find $argv0_dir -maxdepth 1 -perm +0111 -type f -name "$WINE_BIN_NAME" -printf "%h\n" | head -1`
fi
if [ ! -x $WINEBIN/$WINE_BIN_NAME ] ; then
WINEBIN=`find $argv0_dir/../ -maxdepth 1 -perm +0111 -type f -name "$WINE_BIN_NAME" -printf "%h\n" | head -1`
fi
if [ ! -x $WINEBIN/$WINE_BIN_NAME ] ; then
WINE_BIN_NAME=wine
if [ ! -x $WINEBIN/$WINE_BIN_NAME ] ; then
WINEBIN=`find $argv0_dir -maxdepth 1 -perm +0111 -type f -name "$WINE_BIN_NAME" -printf "%h\n" | head -1`
fi
if [ ! -x $WINEBIN/$WINE_BIN_NAME ] ; then
WINEBIN=`find $argv0_dir/../ -maxdepth 1 -perm +0111 -type f -name "$WINE_BIN_NAME" -printf "%h\n" | head -1`
fi
fi
if [ ! -r $WINELIB/libwine.so ] ; then
WINELIB=`find $argv0_dir -maxdepth 2 -name 'libwine.so' -printf "%h\n" | head -1`
fi
if [ ! -r $WINELIB/libwine.so ] ; then
WINELIB=`find $argv0_dir/../ -maxdepth 2 -name 'libwine.so' -printf "%h\n" | head -1`
fi
if [ -x $WINEBIN/wineserver ] ; then
WINESERVER=$WINEBIN/wineserver
fi
#------------------------------------------------------------------------------
# Hey, if we built Wine from source, let's add a little extra fun to
# mix it up a bit
#------------------------------------------------------------------------------
if [ -x $WINEBIN/server/wineserver ] ; then
WINESERVER=$WINEBIN/server/wineserver
fi
if [ -r $WINELIB/dlls/libuser.so ] ; then
WINELIBDLLS=$WINELIB/dlls
fi
#------------------------------------------------------------------------------
# Okay, set the paths and move on.
#------------------------------------------------------------------------------
export LD_LIBRARY_PATH=$WINELIB:$WINELIBDLLS:$LD_LIBRARY_PATH
export PATH=$WINEBIN:$PATH
export WINEDLLPATH=$WINELIBDLLS
export WINELOADER=$WINEBIN/$WINE_BIN_NAME
info_flag=~/.wine/.no_prelaunch_window_flag
debug_flag=~/.wine/.no_debug_window_flag
debug_options="-debugmsg warn+all"
if [ -f $info_flag ] ; then
use_info_message=0
else
use_info_message=1
fi
if [ -f $debug_flag ] ; then
use_debug_message=0
else
use_debug_message=1
fi
#------------------------------------------------------------------------------
# No arguments? Help 'em out
#------------------------------------------------------------------------------
always_see_output=0
no_args=0
if [ $# -eq 0 ] ; then
no_args=1
fi
if [ $# -eq 1 -a foo$1 = foo ] ; then
no_args=1
fi
if [ $no_args -eq 1 ] ; then
echo "Wine called with no arguments."
echo "Invoking $WINEBIN/$WINE_BIN_NAME $@ ..."
$XMESSAGE -buttons " Okay ":0," See the Wine Usage Statement ":1," Configure Wine ":2 \
-title "Welcome to Wine" \
"
You have started Wine without specifying any arguments.
Wine requires a least one argument - the name of the Windows
application you would like to run.
If you have launched this through the KDE menu system,
you can use the KDE file browser to select a Windows
exectuable and then click on it to launch Wine with
that application.
You can similarly use the GNOME file manager to
select a Windows executable and double click on it.
If you would like to see the command line arguments
for Wine, select the second option, below.
"
welcome_rc=$?
if [ $welcome_rc -eq 0 ] ; then
exit
fi
if [ $welcome_rc -eq 2 ] ; then
which winesetup
if [ $? -eq 0 ] ; then
winesetup
else
if [ -x /opt/wine/bin/winesetup ] ; then
/opt/wine/bin/winesetup
else
$XMESSAGE -title "Error" "Error: Unable to find winesetup in your PATH or in /opt/wine/bin"
fi
fi
exit
fi
use_info_message=0
always_see_output=1
fi
#------------------------------------------------------------------------------
# No config file? Offer to help 'em out...
#------------------------------------------------------------------------------
conf=0
while [ $conf -eq 0 ] ; do
if [ -f ~/.winerc ] ; then
conf=1
fi
if [ -f ~/.wine/config ] ; then
conf=2
fi
if [ -f /etc/wine.conf ] ; then
conf=3
fi
if [ $conf -ne 0 ] ; then
break;
fi
echo "No configuration file detected."
$XMESSAGE -buttons " Cancel ":0," Proceed ":1," Configure Wine ":2 \
-title "Welcome to Wine" \
"
You have started Wine but we cannot find a Wine
configuration file.
This is normal if you have never run Wine before.
If this is the case, select the 'Configure Wine'
option, below, to create a configuration file.
"
init_rc=$?
if [ $init_rc -eq 0 ] ; then
exit
fi
if [ $init_rc -eq 1 ] ; then
break
fi
if [ $init_rc -eq 2 ] ; then
which winesetup
if [ $? -eq 0 ] ; then
winesetup
else
if [ -x /opt/wine/bin/winesetup ] ; then
/opt/wine/bin/winesetup
else
$XMESSAGE -title "Error" "Error: Unable to find winesetup in your PATH or in /opt/wine/bin"
fi
fi
fi
done
#------------------------------------------------------------------------------
# Optionally Warn the user we're going to be launching Wine...
#------------------------------------------------------------------------------
if [ $use_info_message -ne 0 ] ; then
echo "Invoking $WINEBIN/$WINE_BIN_NAME $@ ..."
$XMESSAGE -timeout 30 -buttons " Dismiss ":0," Never display this message again ":3 \
-title "Wine Launch Window" \
"Invoking $WINEBIN/$WINE_BIN_NAME $@ ...
This dialog box is a temporary status dialog to let you know
that Wine is attempting to launch your application.
Since Wine is still very much in a development stage, many
applications will fail silently. This dialog box is your indication
that we're *trying* to run your application.
This dialog box will automatically disappear after 30 seconds,
or after your application finishes.
You can permanently disable this dialog by selecting the option below.
" &
info_message_pid=$!
fi
#------------------------------------------------------------------------------
# Here's a little function to clean up after that dialog...
#------------------------------------------------------------------------------
function clean_up_info_message ()
{
if [ $use_info_message -ne 0 ] ; then
#------------------------------------------------------------------------------
# Okay, make sure that the notice window is dead (and kill it if it ain't)
#------------------------------------------------------------------------------
ps $info_message_pid >/dev/null 2>&1
if [ $? -ne 0 ] ; then
wait $info_message_pid
info_return=$?
else
info_return=0
kill $info_message_pid
fi
#------------------------------------------------------------------------------
# If they didn't like the warning window, turn it off
#------------------------------------------------------------------------------
if [ $info_return -eq 3 ] ; then
$XMESSAGE -title "Wine Prelaunch Control" \
"Wine will now disable the prelaunch Window you just saw.
You will no longer be notified when Wine is attempting
to start a Windows application.
You can reenable this Window by removing the $info_flag file." -buttons " Okay ":0," Cancel ":1
if [ $? -eq 0 ] ; then
touch $info_flag
fi
fi
fi
use_info_message=0
}
#------------------------------------------------------------------------------
# Generate a temporary log file name
#------------------------------------------------------------------------------
use_log_name=0
log_name=`mktemp /tmp/wine.log.XXXXXX`
if [ $? -eq 0 ] ; then
which tail >/dev/null 2>&1
if [ $? -eq 0 ]; then
use_log_name=1
fi
fi
#------------------------------------------------------------------------------
# Okay, really launch Wine...
#------------------------------------------------------------------------------
if [ $use_log_name -ne 0 ] ; then
#------------------------------------------------------------------------------
# Okay, we bend over backwards to run Wine, get that status,
# but still display its output to the screen.
# The obvious thing to do is to run wine and pipe output to tee,
# but then I can't find a way to get the return code of wine;
# I only get the return code of tee.
#------------------------------------------------------------------------------
$WINEBIN/$WINE_BIN_NAME "$@" >$log_name 2>&1 &
wine_pid=$!
tail -f $log_name &
tail_pid=$!
wait $wine_pid
wine_return=$?
kill $tail_pid
else
$WINEBIN/$WINE_BIN_NAME "$@"
wine_return=$?
fi
#------------------------------------------------------------------------------
# Test the return code, and see if it fails
#------------------------------------------------------------------------------
if [ $always_see_output -eq 0 -a $wine_return -eq 0 ] ; then
echo "Wine exited with a successful status"
if [ $use_log_name -ne 0 ] ; then
rm -f $log_name
fi
else
if [ $always_see_output -eq 0 ] ; then
echo "Wine failed with return code $wine_return"
fi
#------------------------------------------------------------------------------
# Gracefully display a debug message if they like...
#------------------------------------------------------------------------------
while [ $use_debug_message -gt 0 ] ; do
#------------------------------------------------------------------------------
# Build up the menu of choices they can make...
#------------------------------------------------------------------------------
BUTTONS=' Okay :0'
if [ $use_log_name -ne 0 ] ; then
BUTTONS="$BUTTONS"', View Log :1'
fi
BUTTONS="$BUTTONS"', Debug :2'
BUTTONS="$BUTTONS"', Configure :4'
BUTTONS="$BUTTONS"', Disable :3'
#------------------------------------------------------------------------------
# Build an error message
#------------------------------------------------------------------------------
MESSAGE="
Wine has exited with a failure status of $wine_return.
Wine is still development software, so there can be many
explanations for this problem.
You can choose to run Wine again with a higher level
of debug messages (the debug option, below).
You can attempt to reconfigure Wine to make it work better.
Note that one change you can make that will dramatically
effect Wine's behaviour is to change whether or not
Wine uses a true Windows partition, mounted under Linux,
or whether it uses an empty Windows directory.
The Wine Configuration program can assist you in making
those changes (select Configure, below, for more).
You can disable this message entirely by selecting the
Disable option below."
if [ $always_see_output -ne 0 -a $wine_return -eq 0 ] ; then
MESSAGE="
Wine has exited with a failure status of $wine_return.
You can disable this message entirely by selecting the
Disable option below."
fi
if [ $use_log_name -ne 0 ] ; then
MESSAGE="$MESSAGE
Wine has captured a log of the Wine output in the file $log_name.
You may view this file by selecting View Log, below."
fi
#------------------------------------------------------------------------------
# Display the message
#------------------------------------------------------------------------------
$XMESSAGE -title "Wine Finished With Error" -buttons "$BUTTONS" "$MESSAGE"
debug_return=$?
#------------------------------------------------------------------------------
# Dismiss the other window...
#------------------------------------------------------------------------------
clean_up_info_message
#------------------------------------------------------------------------------
# Process a configure instruction
#------------------------------------------------------------------------------
if [ $debug_return -eq 4 ] ; then
which winesetup
if [ $? -eq 0 ] ; then
winesetup
else
if [ -x /opt/wine/bin/winesetup ] ; then
/opt/wine/bin/winesetup
else
$XMESSAGE -title "Error" "Error: Unable to find winesetup in your PATH or in /opt/wine/bin"
fi
fi
continue;
fi
#------------------------------------------------------------------------------
# Process a view instruction
#------------------------------------------------------------------------------
if [ $debug_return -eq 1 ] ; then
$XMESSAGE -title "View Wine Log" -file $log_name -buttons " Okay ":0,"Delete $log_name":1
if [ $? -eq 1 ] ; then
echo "Deleting $log_name"
rm -f $log_name
use_log_name=0
fi
else
use_debug_message=0
fi
#------------------------------------------------------------------------------
# If they didn't like the warning window, turn it off
#------------------------------------------------------------------------------
if [ $debug_return -eq 3 ] ; then
$XMESSAGE -title "Wine Debug Log Control" \
"Wine will now disable the Wine debug output control window you just saw.
You will no longer be notified when Wine fails to start a
Windows application.
You can reenable this Window by removing the $debug_flag file." -buttons " Okay ":0," Cancel ":1
if [ $? -eq 0 ] ; then
touch $debug_flag
fi
fi
#------------------------------------------------------------------------------
# If they want to retry with debug, let 'em.
#------------------------------------------------------------------------------
if [ $debug_return -eq 2 ] ; then
echo "Rerunning $0 $debug_options $@"
exec $0 $debug_options $@
fi
done
fi
clean_up_info_message