User:BELATHUS/Front End: Difference between revisions
(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]+)['"]>([^&]*)><\/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]+['"]>[^&]*><\/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 04: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.