User:BELATHUS/Front End: Difference between revisions

The official GemStone IV encyclopedia.
< User:BELATHUS
Jump to navigation Jump to search
(Current source code for the project.)
(Removed source code, linking to the site where the source code is currently kept. Added some other information on the client and listed the things that I'm still working on.)
Line 1: Line 1:
The most current source code for this project is now hosted on SourceForge.net. Specifically, http://sourceforge.net/projects/archlich/ is the project page, and one can find the CVS Repository listing at http://archlich.cvs.sourceforge.net/archlich/src/. The project is called, simply, "ArchLich". Information on using CVS can be found here: http://sourceforge.net/cvs/?group_id=199690. The ''modulename'' for the source code is "src".
This article contains the current source code for my project. Keep in mind that while this program runs, it is almost utterly devoid of features and isn't finished in any way. There is no scripting support, no highlighting, nothing.


== ArchLich ==
Currently, running the front end requires Ruby 1.8.5 or later with the Ruby/gtk2
libraries installed. In order to log in to the actual game, run the main.rb script
using the command 'ruby ./main.rb' while in the directory containing the front end.
There is also a test instance which allows running the front end without logging in
to GemStone IV, which can be ran by running the command 'ruby ./test.rb' in the same
directory.


;Things to do
<pre>
* ''(in progress)'' Format text in the TextView port.
#!/usr/bin/env ruby
* ''(in progress)'' Teach program to parse XML.
# This script is written by Andrew Sage.
* ''(in progress)'' Complete functionality of 'Dynamic'-type windows.

* Add spell duration timer to the Spellfront window.
require 'gtk2'
* Create a menu item that allows you to chose which windows are open.
require 'socket'
* Add option to move status bars to a separate window.

* Status indicators.
# Settings!
$localecho = true
;Known bugs

* None, other than the program is incomplete.
# Text tags, for the TagTables.
$tag = Hash.new
;Things recently done
$tag['monbold'] = Gtk::TextTag.new # Monsterbold.
* Create a way to save window sizes and positions between logins.

$_TAGHASH_ = Hash.new

# This class is the login screen window.
class LOGIN_WINDOW # Creates the login window.
def initialize()
# Starting the connection here. Then, displaying the login window.
@window = Gtk::Window.new # Creates the login window.
@window.title = "GemStone IV Login"
@window.signal_connect("delete_event") { Gtk.main_quit; false }

# This window will only use a vertical box. Four elements.
@mainbox = Gtk::VBox.new(false, 0)
# I suppose creating a menu bar for this window is pointless... so I won't.
# The following line is all the major objects in the login window.
accountframe(); passwordframe(); buttons(); characterlist(); statusbar()
#Packing the frames into the main box.
@mainbox.pack_start(@account_frame, true, true, 0)
@mainbox.pack_start(@password_frame, true, true, 0)
@mainbox.pack_start(@character_list, true, true, 0)
@mainbox.pack_start(@character_button, false, false, 0)
@mainbox.pack_start(@login_button, false, false, 0)
@mainbox.pack_start(@status_bar, false, false, 0)
@window.add(@mainbox)
@window.show_all
@character_button.visible = false # I have no idea why this doesn't work before the windows are shown.
end

def accountframe()
@account_frame = Gtk::Frame.new(nil)
@account_label = Gtk::Label.new
@account_label.set_markup("<span foreground='#003300' weight='bold'>Account Name</span>")
@account_frame.label_widget = @account_label
@account_frame.shadow_type = Gtk::SHADOW_NONE # Hides a border.
@account_entry = Gtk::ComboBoxEntry.new(true)
@account_frame.add(@account_entry)
end

def passwordframe()
@password_frame = Gtk::Frame.new(nil)
@password_frame.shadow_type = Gtk::SHADOW_NONE # Hides a border.
@password_label = Gtk::Label.new
@password_label.set_markup("<span foreground='#003300' weight='bold'>Password</span>")
@password_frame.label_widget = @password_label
@password_entry = Gtk::Entry.new()
@password_entry.visibility = false #This makes the frame display the "Password" character.
@password_frame.add(@password_entry)
end

#Defines the buttons and sets up their signals when pressed.
def buttons()
@login_button = Gtk::Button.new("_Login", true)
@character_button = Gtk::Button.new("_Choose", true)
@login_button.signal_connect("clicked") do
if @characters = login(@account_entry.child.text, @password_entry.text)
# This removes the account and password frames and displays the character list.
@password_frame.visible = false
@account_frame.visible = false
@character_list.visible = true
@login_button.visible = false
@character_button.visible = true
@character_hash = Hash.new
@characters.each_index do |n| @characters[n].strip! end
unless @characters[6] == nil
@character_hash["#{@characters[6]}"] = Array.new
@character_hash["#{@characters[6]}"][0] = @characters[6]
@character_hash["#{@characters[6]}"][1] = @characters[5]
object = @character_list_model.append(); object[0] = @characters[6]
n = 8; until @characters[n] == nil
@character_hash["#{@characters[n]}"] = Array.new
@character_hash["#{@characters[n]}"][0] = @characters[n]
@character_hash["#{@characters[n]}"][1] = @characters[n - 1]
object = @character_list_model.append(); object[0] = @characters[n]
n += 2
end
@character_list.append_column(Gtk::TreeViewColumn.new("Name", Gtk::CellRendererText.new, :text => 0))
#@character_list.append_column(Gtk::TreeViewColumn.new("Info", Gtk::CellRendererText.new, :text => 1))
else
@status_bar.push(0, "No characters!")
end
end
end
@character_button.signal_connect("clicked") do
#@window.hide; $main.show
selection = @character_list.selection
if selected = selection.selected then choose(selected[0]) else puts "Error!" end
end
end

def characterlist()
@character_list_model = Gtk::ListStore.new(String)
@character_list = Gtk::TreeView.new(@character_list_model)
@character_list.visible = false
end

def statusbar()
@status_bar = Gtk::Statusbar.new
end

def login(account, password)
$_SERVER_.puts("K")
hashkey = $_SERVER_.gets
password.length.times { |n| password[n] = ((password[n] - 32) ^ hashkey[n] + 32) }
$_SERVER_.puts("A\t#{account}\t#{password}")
check = $_SERVER_.gets
# If login is correct, break loop, else, redo the loop.
if check =~ /\tKEY\t/
$_SERVER_.puts "M"
servers = $_SERVER_.gets
#puts servers
$_SERVER_.puts "N\tGS3"
stormfront = $_SERVER_.gets
# This line tells which services are supported by the given connection.
if stormfront =~ /STORM/
$_SERVER_.puts "F\tGS3"
accountstatus = $_SERVER_.gets
#puts accountstatus
if accountstatus =~ /EXPIRED|NEW_TO_GAME|\?/
@status_bar.push(0, "Login Failed: Expired account")
return false
else
$_SERVER_.puts "G\tGS3" # Gets website address and the like. Toss it.
$_SERVER_.gets
$_SERVER_.puts "P\tGS3" # Gets pricing information. Toss it.
$_SERVER_.gets
$_SERVER_.puts "C" # Sends information about the characters. I want to save this.
return $_SERVER_.gets.split("\t")
end
else
@status_bar.push(0, "Login Failed: Uh... GemStone IV doesn't support the StormFront protocol?")
return false
end
else
@status_bar.push(0, "Login Failed: Account or password incorrect")
return false
end
end

def choose(character)
puts @character_hash[character][1]
$_SERVER_.puts "L\t#{@character_hash[character][1]}\tSTORM"
if (logindata = $_SERVER_.gets) =~ /GAMEHOST/
#puts "Login data received."
@window.destroy
logindata = logindata.strip.split("\t")
gamehost = logindata[7].gsub(/GAMEHOST=/, '')
gameport = logindata[8].gsub(/GAMEPORT=/, '').to_i
gamekey = logindata[9].gsub(/KEY=/, '').strip
#puts "#{gamehost}, #{gameport}, #{gamekey}"
$_SERVER_.close
$_SERVER_ = TCPsocket.open(gamehost, gameport)
$main.new_thread # Starts the "get" loop as a new thread. A part of the main window.
$_SERVER_.puts(gamekey)
$_SERVER_.puts "<c>/FE:WIZARD /VERSION:1.0.1.22 /P:WIN_XP /XML"
sleep 1.5; $_SERVER_.puts "<c>"; sleep 1.5; $_SERVER_.puts "<c>_STATE CHATMODE OFF"
else
puts "Apparantly, this is something else..."
puts logindata
end
end
end

# This particular class is the main game window.
class MAIN_WINDOW # Creates a window with an entry bar.
def initialize()
@window = Gtk::Window.new # Creates the main window.
@window.set_size_request(300, 150)
@window.title = "Story"
@window.signal_connect("delete_event") { send("quit"); false }

#mainbox is the primary vertical box in the main window.
@mainbox = Gtk::VBox.new(false, 0)

#story is the text window in the main window.
@story_scroll = Gtk::ScrolledWindow.new
@story_scroll.set_policy(Gtk::POLICY_NEVER, Gtk::POLICY_ALWAYS)
#Creating the "TextView" object to put inside the "ScrolledWindow" object.
@story = Gtk::TextView.new
@story.editable = false
@story.wrap_mode = Gtk::TextTag::WRAP_WORD
@story_scroll.add(@story)
@story_buffer = @story.buffer

#entry is the text entry bar under the story window in the main window.
@entry = Gtk::Entry.new
# This allows the program to pick up when the ENTER key is pressed in the "entry" bar.
@entry.signal_connect("activate") do
send(@entry.text)
@entry.text = ""
end
@statusbox = Gtk::HBox.new(false, 0) # This will contain the four bars at the bottom (health, stamina, etc.)
@healthbar = Gtk::ProgressBar.new
@healthbar.fraction = 1.0; @healthbar.text = "Health"
redbar = Gtk::Style.new
redbar.set_bg(Gtk::STATE_PRELIGHT, 0xFFFF, 0x0000, 0x0000)
@healthbar.set_style(redbar)
@manabar = Gtk::ProgressBar.new
@manabar.fraction = 1.0; @manabar.text = "Mana"
bluebar = Gtk::Style.new
bluebar.set_bg(Gtk::STATE_PRELIGHT, 0x000, 0x0000, 0xFFFF)
@manabar.set_style(bluebar)
@staminabar = Gtk::ProgressBar.new
@staminabar.fraction = 1.0; @staminabar.text = "Stamina"
yellowbar = Gtk::Style.new
yellowbar.set_bg(Gtk::STATE_PRELIGHT, 0xFFFF, 0xFFFF, 0x0000)
@staminabar.set_style(yellowbar)
@spiritbar = Gtk::ProgressBar.new
@spiritbar.fraction = 1.0; @spiritbar.text = "Spirit"
greybar = Gtk::Style.new
greybar.set_bg(Gtk::STATE_PRELIGHT, 0x9999, 0x9999, 0x9999)
@spiritbar.set_style(greybar)
@statusbox.pack_start(@healthbar, true, true, 0)
@statusbox.pack_start(@manabar, true, true, 0)
@statusbox.pack_start(@staminabar, true, true, 0)
@statusbox.pack_start(@spiritbar, true, true, 0)

@mainbox.pack_start(@story_scroll, true, true, 0)
@mainbox.pack_start(@entry, false, false, 0)
@mainbox.pack_start(@statusbox, false, false, 0)

@window.add(@mainbox)
end
def show
@window.show_all
end

def send(command) # Sends a command to the server and, if $localecho is true, echo it to story window.
if command =~ /^quit/i
echo "Closed connection."
Gtk.main_quit
$_SERVER_.puts("<c>" + command)
$_SERVER_.close
elsif command =~ /^\.\w+/
puts "Scripting is not yet supported."
else
$_SERVER_.puts("<c>" + command)
end
if $localecho == true
scrollbar = @story_scroll.vadjustment
#puts "-------------------------"
#puts "Value: #{scrollbar.value}"
#puts "Page: #{scrollbar.page_size}"
#puts "Upper: #{scrollbar.upper}"
#puts "Combined: #{scrollbar.value + scrollbar.page_size}"
@story_buffer.insert(@story_buffer.end_iter, "#{command}\n") # insert text at end of buffer
scrollbar.value = scrollbar.upper #This puts the scrollbar at the bottom of the window.
end
end

def receive(line) # Echo text to story window and put it in the "most recent line" buffer.
# Should see about saving the number of lines and deleting them as the buffer gets too large.
line.strip!
# puts $suspend_stream #Used for debugging.
n = 0
streams = $stream.keys.join('|')
while true
puts "[[#{line}]]"
if n >= 10 then puts "Broke on over-looping"; puts line; break else n += 1 end
if line =~ /<streamWindow id=['"]main['"] title=["']Story['"] subtitle="([^"]+)"[^>]+>/
subtitle = $1; #puts "Changing main window name...#{subtitle}"
line.gsub!(/<streamWindow id=["']main['"] title=["']Story['"] subtitle=\"([^"]+)"[^>]+>/, '')
@window.title = "Story#{subtitle}"
redo
elsif line =~ /<streamWindow id='room' title='Room' subtitle="([^\"]+)"[^>]+>/
#puts "Changing room title."
$stream['room'].title($1)
line.gsub!(/<streamWindow id='room' title='Room' subtitle="[^\"]+"[^>]+>/, '')
redo
elsif line =~ /<compDef id='([^\']+)'>(.*?)<\/compDef>/
#puts "Caught component definition."
type = $1; descrip = $2; $stream['room'].component(type, descrip)
line.gsub!(/<compDef id='[^\']+'>.*?<\/compDef>/, '')
redo
elsif line =~ /<clear(?:Stream|Container) id=['"](\w+)['"]\/>/
stream = $1
$stream[stream].clearbuffer
line.gsub!(/<clear(?:Stream|Container) id=["']#{stream}['"]\/>/, '')
redo
## STREAMS ##
elsif line =~ /<pushStream id=["'](#{streams})['"]\/>/ or $suspend_stream =~ /#{streams}/i
if line =~ /<pushStream id=["'](#{streams})['"]\/>/
stream = $1
#puts "Handling stream: #{stream}"
linesplit = line.split(/<pushStream id=['"][^'"]+['"]\/>/)
handlestream(stream, linesplit[1])
line = linesplit[0]
redo
else
handlestream($suspend_stream, line)
end
## Handle containers ##
elsif line =~ /<exposeContainer id=['"](stow)["']\/>/
container = $1
puts "Exposed container: #{container}"
$stream[container].show
line.gsub!(/<exposeContainer id=['"][a-z]+["']\/>/, '')
redo
# Container streams
elsif line =~ /<container id='([^\']+)' title="([^\"]+)"/
puts "Container title update"
if line =~ /<container id=["'](#{streams})['"] title="([^\"]+)"[^\/]*\/>/
stream = $1; title = $2; puts "Title: #{title}; Stream: #{stream}"
$stream[stream].title(title)
line.gsub!(/<container id=['"][^'"]+['"] title="[^\"]+"[^\/]*\/>/, '')
redo
end
elsif line =~ /<inv id='(\w+)'>(.+?)<\/inv>/
stream = $1; text = $2
$stream[stream].receive(text)
line.gsub!(/<inv id='\w+'>.+?<\/inv>/, '')
redo
# Unknown streams
elsif line =~ /<pushStream id=['"]([^'"]+)['"]\/>/ or $suspend_stream !~ /main|blank/
if line =~ /<pushStream id=['"]([^'"]+)['"]\/>/
stream = $1
puts "#{stream} is undefined."
linesplit = line.split(/<pushStream id=['"][^'"]+['"]\/>/)
$suspend_stream = stream
puts linesplit[1] # This is to allow capturing of an undefined stream.
line = linesplit[0]
if linesplit[1] =~ /<popStream[^\/]+\/>/
line = linesplit[1].gsub!(/.*<popStream[^\/]+\/>/, '')
$suspend_stream = 'main'
redo
end
redo
else
stream = $suspend_stream
puts line
if line =~ /<popStream[^\/]+\/>/
line.gsub!(/.*<popStream[^\/]+\/>/, '')
$suspend_stream = 'main'
redo
end
end
## Handle the prompt ## Handling the prompt is somewhat unique.
elsif line =~ /<prompt time=['"]([0-9]+)['"]>([^&]*)&gt;<\/prompt>/
prompttime = $1; status = $2
$time_offset = Time.now.to_f - prompttime.to_f
#puts "Local: #{Time.now.to_i}, Remote: #{prompttime.to_i}, Time offset: #{$time_offset}"
line.gsub!(/<prompt time=['"][0-9]+['"]>[^&]*&gt;<\/prompt>/, "#{status}>")
redo
#Handle roundtime. This is what we save the $time_offset for.
elsif line =~ /<roundTime value=['"]([0-9]+)['"]\/>/
roundtime = $1.to_f - Time.now.to_f + $time_offset
puts "Roundtime: #{roundtime} seconds"
Thread.new do
$_TAGHASH["roundtime"] = true
sleep roundtime
$_TAGHASH["roundtime"] = false
end
#get health
elsif line =~ /<progressBar id='health' value='([^']+)' text='([^']+)'[^>]+\/>/
value = $1.to_f / 100.0; text = $2
changehealth(value, text)
line.gsub!(/<progressBar id='health'[^>]+>/, '')
#puts line
redo
#get mana
elsif line =~ /<progressBar id='mana' value='([^']+)' text='([^']+)'[^>]+\/>/
value = $1.to_f / 100.0; text = $2
changemana(value, text)
line.gsub!(/<progressBar id='mana'[^>]+>/, '')
#puts line
redo
#get stamina
elsif line =~ /<progressBar id='stamina' value='([^']+)' text='([^']+)'[^>]+\/>/
value = $1.to_f / 100.0; text = $2
changestamina(value, text)
line.gsub!(/<progressBar id='stamina'[^>]+>/, '')
#puts line
redo
elsif line =~ /<progressBar id='spirit' value='([^']+)' text='([^']+)'[^>]+\/>/
value = $1.to_f / 100.0; text = $2
changespirit(value, text)
line.gsub!(/<progressBar id='spirit'[^>]+>/, '')
#puts line
redo
elsif line =~ /<dialogData id='minivitals'>.+<\/dialogData>/
line.gsub!(/<dialogData id='minivitals'>.+<\/dialogData>/, '')
redo
#<progressBar id='stamina' value='67' text='stamina 61/91' left='50%' customText='t' top='0%' width='25%' height='100%'/>
elsif line =~ /<mode id="GAME"\/>/
$suspend_stream = 'main'
elsif line =~ /<mode id="CMGR"\/>/
$suspend_stream = 'blank'
elsif line =~ /<popStream[^>]*\/>/i
$suspend_stream = 'main'
line.gsub!(/<popStream[^>]*\/>/i, '')
unless line =~ /^$/ then fecho(line) end
puts "Erroneous popStream."
# More special prompt handling. Eventually change the fecho to its own sort of echo.
elsif line =~ /^\w+>$/ then fecho(line)
elsif line =~ /^$/ then nil # If the line ends up blank, do nothing else.
else
fecho(line)
end
break
end
end

def handlestream(stream, line)
# puts "Stream handled: #{stream}"
$suspend_stream = stream
if line =~ /<popStream[^>]*\/>(.*)/
saveline = $1
line.gsub!(/<popStream[^>]*\/>.*/, '')
unless line =~ /^$/ then $stream[stream].receive(line) end
$suspend_stream = 'main'
receive(saveline)
else
$stream[stream].receive(line)
end
end

def fecho(line) # Force echo. Ignores the status of the $localecho variable.
scrollbar = @story_scroll.vadjustment
if scrollbar.upper <= scrollbar.value + scrollbar.page_size + 500
readjust = true
else
readjust = false
end
@story_buffer.insert(@story_buffer.end_iter, "#{line}\n") # inserts text at end of buffer
if readjust then scrollbar.value = scrollbar.upper end
end

def echo(line) # Simply echo text back to the story window, if $localecho is on.
if $localecho == true
fecho(line)
end
end

def changehealth(value, text)
@healthbar.fraction = value; @healthbar.text = text.capitalize
end

def changemana(value, text)
@manabar.fraction = value; @manabar.text = text.capitalize
end

def changestamina(value, text)
@staminabar.fraction = value; @staminabar.text = text.capitalize
end

def changespirit(value, text)
@spiritbar.fraction = value; @spiritbar.text = text.capitalize
end

def new_thread()
@window.show_all
$stream.each_key { |key| $stream[key].show }
$stream['room'].show
$suspend_stream = "blank"
receive("This is working.")
Thread.new do
until $_SERVER_.closed?
line = $_SERVER_.gets
receive(line)
#puts "Received a line."
end
puts "Connection closed."
end
Thread.new do
while true; line = gets; receive(line); end # This allows emulating a server message from the terminal.
end
end
end

class TEXT_WINDOW # Creates a window with only a TextView.
def initialize(title)
@window = Gtk::Window.new # Creates the main window.
@window.set_size_request(300, 150)
@window.title = title
@window.signal_connect("delete_event") { send("quit"); true }
#story is the text window in the main window.
@text_scroll = Gtk::ScrolledWindow.new
@text_scroll.set_policy(Gtk::POLICY_NEVER, Gtk::POLICY_ALWAYS)
#Creating the "TextView" object to put inside the "ScrolledWindow" object.
@text = Gtk::TextView.new
@text.editable = false
@text.wrap_mode = Gtk::TextTag::WRAP_WORD
@text_scroll.add(@text)
@text_buffer = @text.buffer
@window.add(@text_scroll)
end

def send(command) # Sends a command to the server and, if $localecho is true, echo it to story window.
if $localecho == true
scrollbar = @text_scroll.vadjustment
#puts "-------------------------"
#puts "Value: #{scrollbar.value}"
#puts "Page: #{scrollbar.page_size}"
#puts "Upper: #{scrollbar.upper}"
#puts "Combined: #{scrollbar.value + scrollbar.page_size}"
@text_buffer.insert(@text_buffer.end_iter, "Cannot close this, yet.\n")
scrollbar.value = scrollbar.upper #This puts the scrollbar at the bottom of the window.
end
end

def receive(line) # receive game data.
scrollbar = @text_scroll.vadjustment
unless line =~ /^$/ then @text_buffer.insert(@text_buffer.end_iter, "#{line}\n") end
scrollbar.value = scrollbar.upper #This puts the scrollbar at the bottom of the window.
end

def clearbuffer()
@text_buffer.text = ''
end

def show()
@window.show_all
end

def title(title = nil)
if title == nil then @window.title else @window.title = "Room#{title}" end
end
end

class ROOM_WINDOW # Creates a window with only a TextView.
def initialize()
@window = Gtk::Window.new # Creates the main window.
@window.set_size_request(300, 75)
@window.title = "Room"
@window.signal_connect("delete_event") { true }
@descrip = Gtk::TextView.new
@descrip.editable = false
@descrip.wrap_mode = Gtk::TextTag::WRAP_WORD
@objects = Gtk::TextView.new
@objects.editable = false
@objects.wrap_mode = Gtk::TextTag::WRAP_WORD
@pcs = Gtk::TextView.new
@pcs.editable = false
@pcs.wrap_mode = Gtk::TextTag::WRAP_WORD
@compass = Gtk::TextView.new
@compass.editable = false
@compass.wrap_mode = Gtk::TextTag::WRAP_WORD
@roombox = Gtk::VBox.new
@roombox.pack_start(@descrip, false, false, 0)
@roombox.pack_start(@objects, false, false, 0)
@roombox.pack_start(@pcs, false, false, 0)
@roombox.pack_start(@compass, true, true, 0)
@descrip_buffer = @descrip.buffer
@objects_buffer = @objects.buffer
@pcs_buffer = @pcs.buffer
@compass_buffer = @compass.buffer
@window.add(@roombox)
end

def show()
@window.show_all
end

def title(title = nil)
if title == nil then @window.title else @window.title = "Room#{title}" end
end
def receive(line)
puts "The room window received a line. #{line}"
if line =~ /<popStream[^>]*\/>/ then $suspend_stream = 'main' end
end
def clearbuffer()
puts "The room window should be cleared."
end
def send(line)
#puts "The room window was sent a line. #{line}"
end
def component(type, descrip)
#puts "#{type}: #{descrip}"
case type
when /room desc/ then descrip(descrip)
when /room objs/ then objects(descrip)
when /room players/ then pcs(descrip)
when /room exits/ then compass(descrip)
when /sprite/ then puts "Sprite?"
else puts "Unknown component definition."
end
end
def descrip(text)
@descrip_buffer.text = text
end
def objects(text)
@descrip_buffer.text = text
end
def pcs(text)
@pcs_buffer.text = text
end
def compass(text)
@compass_buffer.text = text
end
def clearbuffer()
nil
end
end

# login windows
$login = LOGIN_WINDOW.new

# after login windows
$main = MAIN_WINDOW.new # Main story window.
$stream = Hash.new # Stream windows.
$stream['death'] = TEXT_WINDOW.new("Deaths")
$stream['spellfront'] = TEXT_WINDOW.new("Active Spells")
$stream['bounty'] = TEXT_WINDOW.new("Bounty")
$stream['inv'] = TEXT_WINDOW.new("Inventory")
$stream['logons'] = TEXT_WINDOW.new("Logons")
$stream['room'] = ROOM_WINDOW.new
$stream['stow'] = TEXT_WINDOW.new("Untitled")
# With the way Gtk works with multiple threads, I need to predefine windows in the main thread.
# If a window is defined in a secondary thread, the script hangs. However, windows can be exposed
# in a secondary thread. Thus, windows that will be 'created' in the secondary thread need to be
# predefined.
#$stream['cont1'] = TEXT_WINDOW.new("Untitled") # Not yet used.

$_SERVER_ = TCPsocket.open('eaccess.play.net', 7900)
Gtk.main
</pre>

Revision as of 05:42, 30 June 2007

The most current source code for this project is now hosted on SourceForge.net. Specifically, http://sourceforge.net/projects/archlich/ is the project page, and one can find the CVS Repository listing at http://archlich.cvs.sourceforge.net/archlich/src/. The project is called, simply, "ArchLich". Information on using CVS can be found here: http://sourceforge.net/cvs/?group_id=199690. The modulename for the source code is "src".

ArchLich

Currently, running the front end requires Ruby 1.8.5 or later with the Ruby/gtk2 libraries installed. In order to log in to the actual game, run the main.rb script using the command 'ruby ./main.rb' while in the directory containing the front end. There is also a test instance which allows running the front end without logging in to GemStone IV, which can be ran by running the command 'ruby ./test.rb' in the same directory.

Things to do
  • (in progress) Format text in the TextView port.
  • (in progress) Teach program to parse XML.
  • (in progress) Complete functionality of 'Dynamic'-type windows.
  • Add spell duration timer to the Spellfront window.
  • Create a menu item that allows you to chose which windows are open.
  • Add option to move status bars to a separate window.
  • Status indicators.
Known bugs
  • None, other than the program is incomplete.
Things recently done
  • Create a way to save window sizes and positions between logins.