Author Topic: Trivia Bot  (Read 12591 times)

0 Members and 1 Guest are viewing this topic.

Offline Wart

  • Newbie
  • *
  • Posts: 17
    • View Profile
Re: Trivia Bot
« Reply #15 on: September 19, 2006, 09:06:14 pm »
I'm major stuck on this :-/
I don't know how to write the hash table out to a file since I'm using <String, PlayerScore> as my hashtable..
I'm using the writeObject method in ObjectOutputStream which seems to require that whatever it writes out be serializable.

I'm really lost on where to go from here.. Is there a way to write it out without using writeObject() or another solution?

Offline Chavo

  • x86
  • Hero Member
  • *****
  • Posts: 2219
  • no u
    • View Profile
    • Chavoland
Re: Trivia Bot
« Reply #16 on: September 19, 2006, 11:55:24 pm »
I wouldn't use the ObjectOutputStream, that is generally best for java client / java server things or for serializing things that do need to be persisted as objects :)

check out the PrintWriter class

You'll have to write your own method to parse the data out of your custom Object, but that should be pretty easy.

Offline Wart

  • Newbie
  • *
  • Posts: 17
    • View Profile
Re: Trivia Bot
« Reply #17 on: September 20, 2006, 12:28:58 am »
check out the PrintWriter class

That sound like what I'd used in the past.. Searched through the API for a Stream and Output and such just never hit on PrintWriter.  Should have it fixed and runnin soon :)

Offline Wart

  • Newbie
  • *
  • Posts: 17
    • View Profile
Re: Trivia Bot
« Reply #18 on: September 20, 2006, 01:54:59 am »
Alright.. Fixed it by making PlayerScore class static :)  Seems to be working well :)  Doesn't track streaks or anything right now, but does run trivia :)  Still gets messed up from time to time on hints and answering questions, but if you just let it keep running it'll make itself right in a question or two.
There's certainly plenty to add.. a top 5 list, streak tracking, etc, but as it is it'll do what most people want :)
Here's a jar:
www.witcheshovel.com/WartsStuff/WartTrivia.jar
And the knowledge.txt that I have right now (over 14000 questions):
www.witcheshovel.com/WartsStuff/knowledge.txt
And the current code:
Code: [Select]
import javax.swing.*;
import callback_interfaces.*;
import exceptions.*;
import plugin_interfaces.*;
import java.util.*;
import java.util.regex.*;
import java.io.*;
import java.util.Timer;
import java.util.TimerTask;



/************************************
 *
 * RunTrivia
 * Designed to run a trivia game in a chat channel..
 * Fairly straight forward..
 * Hopefully add score keepin and such..
 * Modeled after the trivia answer bot by Derrenks
 *
 ************************************/


public class PluginMain extends GenericPluginInterface implements CommandCallback, EventCallback
{
private PublicExposedFunctions out;
private static StaticExposedFunctions staticFuncs;

private String BOTNAME = "TriviaBot";
private Vector knowledge; //Contents of knowledge.txt
private String question;
private String answer;
private String oldAnswer = "nothing";
private boolean needReset;
private boolean triviaOn=false;
private boolean questionAsked=false;
private int questionNumber;
private HintThread hintThread;
private Hashtable<String, PlayerScore> scoreTable;
private PlayerScore playerScore;
static private Random gen = new Random();


public void load(StaticExposedFunctions staticFuncs) {
PluginMain.staticFuncs = staticFuncs;
}

public void activate(PublicExposedFunctions out, PluginCallbackRegister register) {
this.out = out;
register.registerCommandPlugin(this, "trivia", 0, false, "ANL", "", "Turns trivia on/off.", null);
register.registerEventPlugin(this, null);
knowledge = new Vector();
loadFile();
loadScores();
}

public void deactivate(PluginCallbackRegister register) {saveScores(); }

public String getName() { return "Wart's Trivia"; }

public String getVersion() { return "0.0.1"; }

public String getAuthorName() { return "Wart"; }

public String getAuthorWebsite() { return "http://www.wartsworld.com"; }

public String getAuthorEmail() { return "wart@wartsworld.com"; }

public String getShortDescription() { return "Runs trivia"; }

public String getLongDescription() { return "Runs a trivia program by picking questions from the specified file in the bot folder"; }

public Properties getDefaultSettingValues() {
//proceeding # because javaop displays them alphabetically...
Properties p = new Properties();
p.setProperty("1. Text to proceed question", ":+:");
p.setProperty("2. Acknowledge correct answers", "%n got it!");
p.setProperty("3. Answer format", "The answer was: ");
p.setProperty("4. Number of hints to provide", "3");
p.setProperty("5. Hint character", "-");
p.setProperty("6. Knowledge file", "knowledge.txt");
p.setProperty("7. Hint Delay", "7000");
p.setProperty("8. Score File", "scores.txt");
return p;
}
public Properties getSettingsDescription()
{
Properties p = new Properties();
p.setProperty("1. Text to proceed question", "Text to put before a question. i.e. ClanBot@USEast: :+:What is the orc builder?");
p.setProperty("2. Acknowledge correct answers", "How to say who answered the question. Use %n to denote the persons name. ie %n got the answer right!");
p.setProperty("3. Answer format", "How to display the correct answer. ie The answer was %a.");
p.setProperty("4. Number of hints to provide", "How many hints to provide before telling the answer.");
p.setProperty("5. Hint character", "Character to substitute in for letters.. ie: Hint: P--n for Peon");
p.setProperty("6. Knowledge file", "Name of file to load questions/answers from. - This may not work correctly");
p.setProperty("7. Hint Delay", "Time to wait between hints (milli seconds)");
p.setProperty("8. Score File", "Name of file to keep scores in.");
return p;
}

public JComponent getComponent(String settingName, String value) {
return null; //all settings are text fields
}

public Properties getGlobalDefaultSettingValues() {
Properties p = new Properties();
return p;
}

public Properties getGlobalSettingsDescription() {
Properties p = new Properties();
return p;
}
public JComponent getGlobalComponent(String settingName, String value) {
return null;
}
public boolean isRequiredPlugin() {
return true; //not sure what this actually does - Me either :-/
}
 
//Open File - Read file into array
private void loadFile() {
String knowledgeFile = getSetting("6. Knowledge file");
if(knowledgeFile == "") return; //probably not needed but...
try {
BufferedReader input = new BufferedReader(new FileReader(new File(knowledgeFile)));
String line;
//format is Question and answer on same line with * inbetween
// i.e Is this a trivia plugin?*Yes
while((line = input.readLine()) != null) {
knowledge.add(line);
}
input.close();
} catch(Exception e) { }
}
//Open Score File
private void loadScores() {
try {
String scoreFile = getSetting("8. Score File");
FileInputStream fileIn = new FileInputStream (scoreFile);
ObjectInputStream objectIn = new ObjectInputStream(new BufferedInputStream(fileIn));
Object object = objectIn.readObject();
scoreTable = (Hashtable)object;
objectIn.close();
fileIn.close();

} catch (Exception e) {createNewScore();}  //score file doesn't exist
}
private void createNewScore() {
scoreTable = new Hashtable<String, PlayerScore>(500);
}
private void saveScores() {
try {
out.showMessage("Hash a a string: " + scoreTable.toString());
String scoreFile = getSetting("8. Score File");
FileOutputStream fileOut = new FileOutputStream (scoreFile);
ObjectOutputStream objectOut = new ObjectOutputStream(fileOut);
Object scores = (Object)scoreTable;
//objectOut.writeObject(scores);
objectOut.writeObject(scoreTable);
objectOut.close();
fileOut.close();
}
catch (NotSerializableException a) {out.showMessage("Not serializable Exeption: " + a);}
catch (Exception e) {out.showMessage("Failed to save file with error:" + e);}
}

//Find question - File must be in format: Question*Answer
public void findQuestion() {
questionNumber = gen.nextInt(knowledge.size());
out.showMessage("Knowledge size is: " + knowledge.size());
// while(questionNumber%2 != 0) {
// questionNumber = gen.nextInt(knowledge.size());
// }
String textLine = (String)knowledge.get(questionNumber);
int asteriskAt = textLine.indexOf('*');
question = textLine.substring(0,asteriskAt);
answer = textLine.substring(asteriskAt+1);
out.showMessage("Answer is: " + answer);
// question = (String)knowledge.get(questionNumber);
// answer = (String)knowledge.get(questionNumber+1);
}

//Ask Question - if trivia on
public void askQuestion() throws IOException, PluginException {
if(!triviaOn) return;
if(!questionAsked) {
out.sendText(getSetting("1. Text to proceed question") + " Q#" + questionNumber + " " + question);
questionAsked=true;
}
//Hint thread
int tempNumber = questionNumber;
int hintMax = Integer.parseInt(getSetting("4. Number of hints to provide"));
int currentHint=0;
char[] hint = new char[answer.length()];
int hintDelay = Integer.parseInt(getSetting("7. Hint Delay"));
char hintChar = getSetting("5. Hint character").charAt(0);
hintThread = new HintThread(hintDelay,hintMax,answer.length(),hintChar);
hintThread.start();
}

//Listen for correct answers
public void talk(String user, String text, int ping, int flags) throws IOException, PluginException {
if(!triviaOn || !questionAsked) return; //Trivia not on or no question asked
if(needReset && oldAnswer == answer) {  //No answer
needReset=false;
resetFromHint();
}
else if(text.equalsIgnoreCase(answer) && questionAsked) { //Correct answer
oldAnswer=answer;
questionAsked=false;
hintThread = null;
int score = addToScore(user);
out.sendTextPriority(user + " got it right and has now answered " + score + " questions! The answer was: " + answer,10);
resetFromHint();
}
}


//Correct Answer --> Keep Score
private int addToScore(String guesser) {
//See if person is already in hash
if(scoreTable.containsKey(guesser)) {
//if so add to their score
playerScore = scoreTable.get(guesser);
playerScore.incTotalCorrect();
playerScore.incLongestStreak();
scoreTable.put(guesser,playerScore);
return playerScore.getTotalCorrect();
}
else {
//if not add them to hash
playerScore = new PlayerScore(guesser);
playerScore.incLongestStreak();
playerScore.incTotalCorrect();
scoreTable.put(guesser, playerScore);
return playerScore.getTotalCorrect();
}
}


public String getAnswer() {
return answer;
}
public int getRandom(int randMax) {
return gen.nextInt(randMax);
}
public void sayOutLoud(String text) {
try{ out.sendText(text); }
catch (Exception e) {}
}
public void resetFromHint() {
if(!triviaOn) return;
oldAnswer = answer;
findQuestion();
try {Thread.sleep(5000);}
catch (Exception e2) {out.showMessage("Didn't sleep after reset");}
try {askQuestion(); }
catch (Exception e) {}
}
public void sayToConsole(String text) {
try{ out.showMessage(text);}
catch (Exception e) {}
}


public void emote(String user, String text, int ping, int flags) throws IOException, PluginException {}
public void whisperFrom(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void whisperTo(String user, String statstring, int ping, int flags) throws IOException, PluginException {}

public void userShow(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void userJoin(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void userLeave(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void userFlags(String user, String statstring, int ping, int flags) throws IOException, PluginException {}

public void error(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void info(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void broadcast(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void channel(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
private String getSetting(String name) {
return out.getLocalSetting(getName(), name);
}

public void commandExecuted(String user, String command, String[] args, int loudness, Object data) throws PluginException, IOException {
if (command.equalsIgnoreCase("trivia")) {
if(triviaOn) {
triviaOn=false;
out.sendText("Trivia Turned Off");
questionAsked = false;
}
else {
triviaOn=true;
out.sendText("Trivia Turned On");
findQuestion();
askQuestion();
}
}
}

private class HintThread extends Thread {
private int hintDelay,hintMax,answerLength;
private char hintChar;
public HintThread(int hintDelay,int hintMax,int answerLength, char hintChar) {
setPriority(MIN_PRIORITY);
this.hintDelay = hintDelay;
this.hintMax = hintMax;
this.answerLength = answerLength;
this.hintChar = hintChar;
this.setName("Hint-thread");
}

public void run() {
if(!triviaOn || !questionAsked) return; //Does this work in a thread?

// setPriority(MIN_PRIORITY);
int currentHint;
char[] hint = new char[answerLength];
for(int i=0;i<answerLength;i++) {
hint[i] = hintChar; //Make hint word all ----
}
try{
for(currentHint=0;currentHint<hintMax;currentHint++)
{
Thread.sleep(hintDelay);
if(questionAsked==false) return;
if(oldAnswer==answer) return;
//give hint - 1/4(?) of the letters - would be nice to make this configurable
//This is done by getting a random char of the string and replacing it with the correct one
for(int j=0;j<(answerLength/3);j++) {
int temp = gen.nextInt(answerLength);
hint[temp] = answer.charAt(temp);
}
out.sendTextPriority("Hint: " + String.valueOf(hint),PRIORITY_NORMAL);
}
Thread.sleep(hintDelay);
if(oldAnswer==answer) return;
if(questionAsked==false) return;
questionAsked=false;
out.sendTextPriority("The answer is: " + answer,10);
// needReset=true;
out.showMessage("hintDelay is: " + hintDelay);
Thread.sleep(hintDelay);
resetFromHint();
}catch(Exception e){}
}
}
public static class PlayerScore implements Serializable { //Class for keeping player scores

String playerName;
int longestStreak=0;
int totalCorrect=0;

public PlayerScore() {}
public PlayerScore(String name) {
playerName = name;
}
public void setTotalCorrect(int newCorrect) {
totalCorrect = newCorrect;
}
public void incTotalCorrect() {
totalCorrect++;
}
public void setLongestStreak(int streak) {
longestStreak=streak;
}
public void incLongestStreak() {
longestStreak++;
}
public String getName() {
return playerName;
}
public int getStreak() {
return longestStreak;
}
public int getTotalCorrect() {
return totalCorrect;
}
}
}



Offline Chavo

  • x86
  • Hero Member
  • *****
  • Posts: 2219
  • no u
    • View Profile
    • Chavoland
Re: Trivia Bot
« Reply #19 on: September 20, 2006, 02:17:48 am »
are you sure you know what static does?

Offline Wart

  • Newbie
  • *
  • Posts: 17
    • View Profile
Re: Trivia Bot
« Reply #20 on: September 20, 2006, 11:18:23 am »
"Static Nested Classes
Finally, Java has an obscure feature where you can also declare nested classes static. This does not mean all their methods are static, or that they cannot be instantiated, rather it means you instantiate the inner classes independently of the main enclosing class. There is no associated outer class object. Such classes are often called nested static classes. Non-static inner class objects always have an associated outer class object."

From:http://mindprod.com/jgloss/static.html
.. Not that that's any sort of great source, but it explains the 'staticness' of the class better than I could :-/

Offline Chavo

  • x86
  • Hero Member
  • *****
  • Posts: 2219
  • no u
    • View Profile
    • Chavoland
Re: Trivia Bot
« Reply #21 on: September 20, 2006, 12:43:46 pm »
I guess that will work actually.  I still recommend not using ObjectOutputStream for things that don't need it though.

Offline Wart

  • Newbie
  • *
  • Posts: 17
    • View Profile
Re: Trivia Bot
« Reply #22 on: September 20, 2006, 04:04:08 pm »
I guess that will work actually.  I still recommend not using ObjectOutputStream for things that don't need it though.

Agreed :)
It's certainly not the cleanest, prettiest code.. but it works :-/

Offline Joe

  • B&
  • Moderator
  • Hero Member
  • *****
  • Posts: 10319
  • In Soviet Russia, text read you!
    • View Profile
    • Github
Re: Trivia Bot
« Reply #23 on: September 22, 2006, 11:50:26 pm »
The way I learned Visual Basic pretty good (I was referred to as God by my classmates) was by writing my code (namely a work of art called JoeBot) over and over again, each time getting better at what I did. If you get bored, scrap this completely and rewrite it from the ground up. I can almost promise you that you'll do at least something better. :)
I'd personally do as Joe suggests

You might be right about that, Joe.


Offline Wart

  • Newbie
  • *
  • Posts: 17
    • View Profile
Re: Trivia Bot
« Reply #24 on: September 23, 2006, 01:01:13 am »
I'll rewrite it when jo3 finally gets released ;-)
Jar here: http://www.wartsworld.com/WartTrivia.jar
knowledge.txt http://www.wartsworld.com/Diablo/zxcvfd/knowledge.txt
Here's the latest code.  It now supports listing the top 3 answerers and keeps track of streaks, though if no one answers a question it doesnt clear the current streak.  commands are .trivia .top3 and .streak

Code: [Select]
import javax.swing.*;
import callback_interfaces.*;
import exceptions.*;
import plugin_interfaces.*;
import java.util.*;
import java.util.regex.*;
import java.io.*;
import java.util.Timer;
import java.util.TimerTask;



/************************************
 *
 * RunTrivia
 * Designed to run a trivia game in a chat channel..
 * Fairly straight forward..
 * Hopefully add score keepin and such..
 * Modeled after the trivia answer bot by Derrenks
 *
 ************************************/


public class PluginMain extends GenericPluginInterface implements CommandCallback, EventCallback
{
private PublicExposedFunctions out;
private static StaticExposedFunctions staticFuncs;

private String BOTNAME = "TriviaBot";
private Vector knowledge; //Contents of knowledge.txt
private volatile String question;
private volatile String answer;
private volatile String oldAnswer = "nothing";
private String oldGuesser = null;
private int streak = 0;
private boolean needReset;
private boolean incStreak = false;
private volatile boolean triviaOn=false;
private volatile boolean questionAsked=false;
private int questionNumber;
private volatile HintThread hintThread;
private Hashtable<String, PlayerScore> scoreTable;
private PlayerScore playerScore;
static private Random gen = new Random();


public void load(StaticExposedFunctions staticFuncs) {
PluginMain.staticFuncs = staticFuncs;
}

public void activate(PublicExposedFunctions out, PluginCallbackRegister register) {
this.out = out;
register.registerCommandPlugin(this, "trivia", 0, false, "ANL", "", "Turns trivia on/off.", null);
register.registerCommandPlugin(this, "top3", 0, false, "ANL", "", "Prints top 3 scores", null);
register.registerCommandPlugin(this, "streak", 0, false, "ANL", "", "Prints Longest 3 Streaks", null);
register.registerEventPlugin(this, null);
knowledge = new Vector();
loadFile();
loadScores();
}

public void deactivate(PluginCallbackRegister register) {saveScores(); }

public String getName() { return "Wart's Trivia"; }

public String getVersion() { return "0.0.1"; }

public String getAuthorName() { return "Wart"; }

public String getAuthorWebsite() { return "http://www.wartsworld.com"; }

public String getAuthorEmail() { return "wart@wartsworld.com"; }

public String getShortDescription() { return "Runs trivia"; }

public String getLongDescription() { return "Runs a trivia program by picking questions from the specified file in the bot folder"; }

public Properties getDefaultSettingValues() {
//proceeding # because javaop displays them alphabetically...
Properties p = new Properties();
p.setProperty("1. Text to proceed question", ":+:");
p.setProperty("2. Acknowledge correct answers", "%n got it!");
p.setProperty("3. Answer format", "The answer was: ");
p.setProperty("4. Number of hints to provide", "3");
p.setProperty("5. Hint character", "-");
p.setProperty("6. Knowledge file", "knowledge.txt");
p.setProperty("7. Hint Delay", "7000");
p.setProperty("8. Score File", "scores.txt");
return p;
}
public Properties getSettingsDescription()
{
Properties p = new Properties();
p.setProperty("1. Text to proceed question", "Text to put before a question. i.e. ClanBot@USEast: :+:What is the orc builder?");
p.setProperty("2. Acknowledge correct answers", "How to say who answered the question. Use %n to denote the persons name. ie %n got the answer right!");
p.setProperty("3. Answer format", "How to display the correct answer. ie The answer was %a.");
p.setProperty("4. Number of hints to provide", "How many hints to provide before telling the answer.");
p.setProperty("5. Hint character", "Character to substitute in for letters.. ie: Hint: P--n for Peon");
p.setProperty("6. Knowledge file", "Name of file to load questions/answers from. - This may not work correctly");
p.setProperty("7. Hint Delay", "Time to wait between hints (milli seconds)");
p.setProperty("8. Score File", "Name of file to keep scores in.");
return p;
}

public JComponent getComponent(String settingName, String value) {
return null; //all settings are text fields
}

public Properties getGlobalDefaultSettingValues() {
Properties p = new Properties();
return p;
}

public Properties getGlobalSettingsDescription() {
Properties p = new Properties();
return p;
}
public JComponent getGlobalComponent(String settingName, String value) {
return null;
}
public boolean isRequiredPlugin() {
return true; //not sure what this actually does -> Me either :-/
}
 
//Open File - Read file into array
private void loadFile() {
String knowledgeFile = getSetting("6. Knowledge file");
if(knowledgeFile == "") return; //probably not needed but...
try {
BufferedReader input = new BufferedReader(new FileReader(new File(knowledgeFile)));
String line;
//format is Question and answer on same line with * inbetween
// i.e Is this a trivia plugin?*Yes
while((line = input.readLine()) != null) {
knowledge.add(line);
}
input.close();
} catch(Exception e) { }
}
//Open Score File
private void loadScores() {
try {
String scoreFile = getSetting("8. Score File");
FileInputStream fileIn = new FileInputStream (scoreFile);
ObjectInputStream objectIn = new ObjectInputStream(new BufferedInputStream(fileIn));
Object object = objectIn.readObject();
scoreTable = (Hashtable)object;
objectIn.close();
fileIn.close();

} catch (Exception e) {createNewScore();}  //score file doesn't exist
}
private void createNewScore() {
scoreTable = new Hashtable<String, PlayerScore>(500);
}
private void saveScores() {
try {
out.showMessage("Hash a a string: " + scoreTable.toString());
String scoreFile = getSetting("8. Score File");
FileOutputStream fileOut = new FileOutputStream (scoreFile);
ObjectOutputStream objectOut = new ObjectOutputStream(fileOut);
Object scores = (Object)scoreTable;
//objectOut.writeObject(scores);
objectOut.writeObject(scoreTable);
objectOut.close();
fileOut.close();
}
catch (NotSerializableException a) {out.showMessage("Not serializable Exeption: " + a);}
catch (Exception e) {out.showMessage("Failed to save file with error:" + e);}
}

//Find question - File must be in format: Question*Answer
public void findQuestion() {
questionNumber = gen.nextInt(knowledge.size());
out.showMessage("Knowledge size is: " + knowledge.size());
// while(questionNumber%2 != 0) {
// questionNumber = gen.nextInt(knowledge.size());
// }
String textLine = (String)knowledge.get(questionNumber);
int asteriskAt = textLine.indexOf('*');
question = textLine.substring(0,asteriskAt);
answer = textLine.substring(asteriskAt+1);
out.showMessage("Answer is: " + answer);
// question = (String)knowledge.get(questionNumber);
// answer = (String)knowledge.get(questionNumber+1);
}

//Ask Question - if trivia on
public void askQuestion() throws IOException, PluginException {
if(!triviaOn) return;
if(!questionAsked) {
out.sendText(getSetting("1. Text to proceed question") + " Q#" + questionNumber + " " + question);
questionAsked=true;
}
//Hint thread
int tempNumber = questionNumber;
int hintMax = Integer.parseInt(getSetting("4. Number of hints to provide"));
int currentHint=0;
char[] hint = new char[answer.length()];
int hintDelay = Integer.parseInt(getSetting("7. Hint Delay"));
char hintChar = getSetting("5. Hint character").charAt(0);
hintThread = new HintThread(hintDelay,hintMax,answer.length(),hintChar);
hintThread.start();
}

//Listen for correct answers
public void talk(String user, String text, int ping, int flags) throws IOException, PluginException {
if(!triviaOn || !questionAsked) return; //Trivia not on or no question asked
if(needReset && oldAnswer == answer) {  //No answer
needReset=false;
resetFromHint();
}
else if(text.equalsIgnoreCase(answer) && questionAsked) { //Correct answer
oldAnswer=answer;
questionAsked=false;
if(oldGuesser != null && oldGuesser.equalsIgnoreCase(user)) {
out.showMessage("oldGuesser == user - oldGuesser = " + oldGuesser);
streak++;
incStreak = true;
}
else {
out.showMessage("oldGuesser != user -> oldGuesser = " + oldGuesser);
out.showMessage("oldGuesser != user -> user = " + user);
streak=1;
oldGuesser = user;
incStreak = false;
}
int score = addToScore(user);
out.sendTextPriority(user + " got it! Score: " + score + " The answer was: " + answer,10);
resetFromHint();
}
}


//Correct Answer --> Keep Score
private int addToScore(String guesser) {
//See if person is already in hash
if(scoreTable.containsKey(guesser)) {
//if so add to their score
playerScore = scoreTable.get(guesser);
playerScore.incTotalCorrect();
scoreTable.put(guesser,playerScore);
//See if they are on a streak
if(incStreak) {
if(streak > playerScore.getStreak()) {
playerScore.incLongestStreak();
out.showMessage(guesser + " streak is: " + playerScore.getStreak());
}
}
return playerScore.getTotalCorrect();
}
else {
//if not add them to hash
playerScore = new PlayerScore(guesser);
playerScore.incLongestStreak();
playerScore.incTotalCorrect();
scoreTable.put(guesser, playerScore);
return playerScore.getTotalCorrect();
}
}


public String getAnswer() {
return answer;
}
public int getRandom(int randMax) {
return gen.nextInt(randMax);
}
public void sayOutLoud(String text) {
try{ out.sendText(text); }
catch (Exception e) {}
}
public void resetFromHint() {
if(!triviaOn) return;
hintThread.quit();
hintThread = null;
oldAnswer = answer;
findQuestion();
try {Thread.sleep(5000);}
catch (Exception e2) {out.showMessage("Didn't sleep after reset");}
try {askQuestion(); }
catch (Exception e) {}
}
public void sayToConsole(String text) {
try{ out.showMessage(text);}
catch (Exception e) {}
}


public void emote(String user, String text, int ping, int flags) throws IOException, PluginException {}
public void whisperFrom(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void whisperTo(String user, String statstring, int ping, int flags) throws IOException, PluginException {}

public void userShow(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void userJoin(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void userLeave(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void userFlags(String user, String statstring, int ping, int flags) throws IOException, PluginException {}

public void error(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void info(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void broadcast(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void channel(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
private String getSetting(String name) {
return out.getLocalSetting(getName(), name);
}

public void commandExecuted(String user, String command, String[] args, int loudness, Object data) throws PluginException, IOException {
if (command.equalsIgnoreCase("trivia")) {
if(triviaOn) {
triviaOn=false;
out.sendText("Trivia Turned Off");
questionAsked = false;
}
else {
triviaOn=true;
out.sendText("Trivia Turned On");
findQuestion();
askQuestion();
}
}
if (command.equalsIgnoreCase("top3")) {
//Get a vector of all players in class
Vector players = new Vector(scoreTable.keySet());
Vector scores = new Vector();
for(int j=0;j<players.size();j++) {
scores.add(scoreTable.get(players.elementAt(j)).getTotalCorrect());
}
Vector sortedScores = (Vector)scores.clone();
Collections.sort(sortedScores);
int sortedSize = sortedScores.size();
    for(int i=0;i<scores.size();i++) {
//if first place
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-1)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-2)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-3)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
    }
}
if (command.equalsIgnoreCase("streak")) {
//Get a vector of all players in class
Vector players = new Vector(scoreTable.keySet());
Vector scores = new Vector();
for(int j=0;j<players.size();j++) {
scores.add(scoreTable.get(players.elementAt(j)).getStreak());
}
Vector sortedScores = (Vector)scores.clone();
Collections.sort(sortedScores);
int sortedSize = sortedScores.size();
    for(int i=0;i<scores.size();i++) {
//if first place
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-1)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-2)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-3)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
    }
}
}

private class HintThread extends Thread {
private int hintDelay,hintMax,answerLength;
private char hintChar;
private volatile Thread thisThread;
public HintThread(int hintDelay,int hintMax,int answerLength, char hintChar) {
setPriority(MIN_PRIORITY);
this.hintDelay = hintDelay;
this.hintMax = hintMax;
this.answerLength = answerLength;
this.hintChar = hintChar;
this.setName("Hint-thread");
}
public void quit() {
thisThread = null;
  }
public void run() {
thisThread = Thread.currentThread();
if(!triviaOn || !questionAsked) return; //Does this work in a thread?

// setPriority(MIN_PRIORITY);
int currentHint;
char[] hint = new char[answerLength];
for(int i=0;i<answerLength;i++) {
hint[i] = hintChar; //Make hint word all ----
}
try{
for(currentHint=0;currentHint<hintMax;currentHint++)
{
Thread.sleep(hintDelay);
//give hint - 1/4(?) of the letters - would be nice to make this configurable
//This is done by getting a random char of the string and replacing it with the correct one
for(int j=0;j<(answerLength/3);j++) {
int temp = gen.nextInt(answerLength);
hint[temp] = answer.charAt(temp);
}
if(thisThread == null  || questionAsked == false || oldAnswer == answer) {out.showMessage("I should be returning"); return;}
else{
out.sendTextPriority("Hint: " + String.valueOf(hint),PRIORITY_NORMAL);
}
}
Thread.sleep(hintDelay);
if(thisThread == null  || questionAsked == false || oldAnswer == answer) {out.showMessage("I should be returning"); return;}
else {
out.sendTextPriority("The answer is: " + answer,10);
}
questionAsked=false;
out.showMessage("hintDelay is: " + hintDelay);
resetFromHint();
}catch(Exception e){}
}
}
public static class PlayerScore implements Serializable { //Class for keeping player scores

String playerName;
int longestStreak=0;
int totalCorrect=0;

public PlayerScore() {}
public PlayerScore(String name) {
playerName = name;
}
public void setTotalCorrect(int newCorrect) {
totalCorrect = newCorrect;
}
public void incTotalCorrect() {
totalCorrect++;
}
public void setLongestStreak(int streak) {
longestStreak=streak;
}
public void incLongestStreak() {
longestStreak++;
}
public String getName() {
return playerName;
}
public int getStreak() {
return longestStreak;
}
public int getTotalCorrect() {
return totalCorrect;
}
}
}



Offline Blaze

  • x86
  • Hero Member
  • *****
  • Posts: 7136
  • Canadian
    • View Profile
    • Maide
Re: Trivia Bot
« Reply #25 on: September 23, 2006, 01:14:43 am »
Good luck on JavaOp3.  :)
And like a fool I believed myself, and thought I was somebody else...

Offline Wart

  • Newbie
  • *
  • Posts: 17
    • View Profile
Re: Trivia Bot
« Reply #26 on: October 29, 2006, 02:22:58 pm »
Here's my latest update.. This works *MUCH* better than any other time and I believe is actually worthy of using :)  Rarely has trouble anymore and runs all the time on the server I play on.  I changed the way hints are given, so that now instead of giving semi random characters it now gives the first 4th of the word, then half, then 3/4ths of the word.  This should help making answering questions easier and keep the pace of the game up.
Only possible thing I might change is saving scores more often than when you close the bot.. Do you think it'd be reasonable to save scores after every answer?  Or maybe implement a counter that saves every 10 answers or something?
You can get this version here: http://www.wartsworld.com/Bot/WartTrivia.jar
Code: [Select]
import javax.swing.*;
import callback_interfaces.*;
import exceptions.*;
import plugin_interfaces.*;
import java.util.*;
import java.util.regex.*;
import java.io.*;
import java.util.Timer;
import java.util.TimerTask;



/************************************
 *
 * RunTrivia
 * Designed to run a trivia game in a chat channel..
 * Fairly straight forward..
 * Hopefully add score keepin and such..
 * Modeled after the trivia answer bot by Derrenks
 *
 ************************************/


public class PluginMain extends GenericPluginInterface implements CommandCallback, EventCallback
{
private PublicExposedFunctions out;
private static StaticExposedFunctions staticFuncs;

private String BOTNAME = "TriviaBot";
private Vector knowledge; //Contents of knowledge.txt
private volatile String question;
private volatile String answer;
private volatile String oldAnswer = "nothing";
private String oldGuesser = null;
private int streak = 0;
private boolean needReset;
private boolean incStreak = false;
private volatile boolean triviaOn=false;
private volatile boolean questionAsked=false;
private int questionNumber;
private volatile HintThread hintThread;
private Hashtable<String, PlayerScore> scoreTable;
private PlayerScore playerScore;
static private Random gen = new Random();


public void load(StaticExposedFunctions staticFuncs) {
PluginMain.staticFuncs = staticFuncs;
}

public void activate(PublicExposedFunctions out, PluginCallbackRegister register) {
this.out = out;
register.registerCommandPlugin(this, "trivia", 0, false, "ANL", "", "Turns trivia on/off.", null);
register.registerCommandPlugin(this, "top3", 0, false, "ANL", "", "Prints top 3 scores", null);
register.registerCommandPlugin(this, "streak", 0, false, "ANL", "", "Prints Longest 3 Streaks", null);
register.registerEventPlugin(this, null);
knowledge = new Vector();
loadFile();
loadScores();
}

public void deactivate(PluginCallbackRegister register) {saveScores(); }

public String getName() { return "Wart's Trivia"; }

public String getVersion() { return "0.0.1"; }

public String getAuthorName() { return "Wart"; }

public String getAuthorWebsite() { return "http://www.wartsworld.com"; }

public String getAuthorEmail() { return "wart@wartsworld.com"; }

public String getShortDescription() { return "Runs trivia"; }

public String getLongDescription() { return "Runs a trivia program by picking questions from the specified file in the bot folder"; }

public Properties getDefaultSettingValues() {
//proceeding # because javaop displays them alphabetically...
Properties p = new Properties();
p.setProperty("1. Text to proceed question", ":+:");
p.setProperty("2. Acknowledge correct answers", "%n got it!");
p.setProperty("3. Answer format", "The answer was: ");
p.setProperty("4. Number of hints to provide", "3");
p.setProperty("5. Hint character", "-");
p.setProperty("6. Knowledge file", "knowledge.txt");
p.setProperty("7. Hint Delay", "7000");
p.setProperty("8. Score File", "scores.txt");
return p;
}
public Properties getSettingsDescription()
{
Properties p = new Properties();
p.setProperty("1. Text to proceed question", "Text to put before a question. i.e. ClanBot@USEast: :+:What is the orc builder?");
p.setProperty("2. Acknowledge correct answers", "How to say who answered the question. Use %n to denote the persons name. ie %n got the answer right!");
p.setProperty("3. Answer format", "How to display the correct answer. ie The answer was %a.");
p.setProperty("4. Number of hints to provide", "How many hints to provide before telling the answer.");
p.setProperty("5. Hint character", "Character to substitute in for letters.. ie: Hint: P--n for Peon");
p.setProperty("6. Knowledge file", "Name of file to load questions/answers from. - This may not work correctly");
p.setProperty("7. Hint Delay", "Time to wait between hints (milli seconds)");
p.setProperty("8. Score File", "Name of file to keep scores in.");
return p;
}

public JComponent getComponent(String settingName, String value) {
return null; //all settings are text fields
}

public Properties getGlobalDefaultSettingValues() {
Properties p = new Properties();
return p;
}

public Properties getGlobalSettingsDescription() {
Properties p = new Properties();
return p;
}
public JComponent getGlobalComponent(String settingName, String value) {
return null;
}
public boolean isRequiredPlugin() {
return true; //not sure what this actually does -> Me either :-/
}
 
//Open File - Read file into array
private void loadFile() {
String knowledgeFile = getSetting("6. Knowledge file");
if(knowledgeFile == "") return; //probably not needed but...
try {
BufferedReader input = new BufferedReader(new FileReader(new File(knowledgeFile)));
String line;
//format is Question and answer on same line with * inbetween
// i.e Is this a trivia plugin?*Yes
while((line = input.readLine()) != null) {
knowledge.add(line);
}
input.close();
} catch(Exception e) { }
}
//Open Score File
private void loadScores() {
try {
String scoreFile = getSetting("8. Score File");
FileInputStream fileIn = new FileInputStream (scoreFile);
ObjectInputStream objectIn = new ObjectInputStream(new BufferedInputStream(fileIn));
Object object = objectIn.readObject();
scoreTable = (Hashtable)object;
objectIn.close();
fileIn.close();

} catch (Exception e) {createNewScore();}  //score file doesn't exist
}
private void createNewScore() {
scoreTable = new Hashtable<String, PlayerScore>(500);
}
private void saveScores() {
try {
out.showMessage("Hash a a string: " + scoreTable.toString());
String scoreFile = getSetting("8. Score File");
FileOutputStream fileOut = new FileOutputStream (scoreFile);
ObjectOutputStream objectOut = new ObjectOutputStream(fileOut);
Object scores = (Object)scoreTable;
//objectOut.writeObject(scores);
objectOut.writeObject(scoreTable);
objectOut.close();
fileOut.close();
}
catch (NotSerializableException a) {out.showMessage("Not serializable Exeption: " + a);}
catch (Exception e) {out.showMessage("Failed to save file with error:" + e);}
}

//Find question - File must be in format: Question*Answer
public void findQuestion() {
questionNumber = gen.nextInt(knowledge.size());
out.showMessage("Knowledge size is: " + knowledge.size());
// while(questionNumber%2 != 0) {
// questionNumber = gen.nextInt(knowledge.size());
// }
String textLine = (String)knowledge.get(questionNumber);
int asteriskAt = textLine.indexOf('*');
question = textLine.substring(0,asteriskAt);
answer = textLine.substring(asteriskAt+1);
if(question.length() > 150) {
out.showMessage("Question was too long");
findQuestion();
}
out.showMessage("Answer is: " + answer);
// question = (String)knowledge.get(questionNumber);
// answer = (String)knowledge.get(questionNumber+1);
}

//Ask Question - if trivia on
public void askQuestion() throws IOException, PluginException {
if(!triviaOn) return;
if(!questionAsked) {
out.sendText(getSetting("1. Text to proceed question") + " Q#" + questionNumber + " " + question);
questionAsked=true;
}
//Hint thread
int tempNumber = questionNumber;
int hintMax = Integer.parseInt(getSetting("4. Number of hints to provide"));
int currentHint=0;
char[] hint = new char[answer.length()];
int hintDelay = Integer.parseInt(getSetting("7. Hint Delay"));
char hintChar = getSetting("5. Hint character").charAt(0);
hintThread = new HintThread(hintDelay,hintMax,answer.length(),hintChar);
hintThread.start();
}

//Listen for correct answers
public void talk(String user, String text, int ping, int flags) throws IOException, PluginException {
if(!triviaOn || !questionAsked) return; //Trivia not on or no question asked
if(needReset && oldAnswer == answer) {  //No answer
needReset=false;
resetFromHint();
}
else if(text.equalsIgnoreCase(answer) && questionAsked) { //Correct answer
oldAnswer=answer;
questionAsked=false;
if(oldGuesser != null && oldGuesser.equalsIgnoreCase(user)) {
streak++;
incStreak = true;
}
else {
streak=1;
oldGuesser = user;
incStreak = false;
}
int score = addToScore(user);
out.sendTextPriority(user + " got it! Score: " + score + " The answer was: " + answer,10);
resetFromHint();
}
}


//Correct Answer --> Keep Score
private int addToScore(String guesser) {
//See if person is already in hash
if(scoreTable.containsKey(guesser)) {
//if so add to their score
playerScore = scoreTable.get(guesser);
playerScore.incTotalCorrect();
scoreTable.put(guesser,playerScore);
//See if they are on a streak
if(incStreak) {
if(streak > playerScore.getStreak()) {
playerScore.incLongestStreak();
out.showMessage(guesser + " streak is: " + playerScore.getStreak());
}
}
return playerScore.getTotalCorrect();
}
else {
//if not add them to hash
playerScore = new PlayerScore(guesser);
playerScore.incLongestStreak();
playerScore.incTotalCorrect();
scoreTable.put(guesser, playerScore);
return playerScore.getTotalCorrect();
}
}


public String getAnswer() {
return answer;
}
public int getRandom(int randMax) {
return gen.nextInt(randMax);
}
public void sayOutLoud(String text) {
try{ out.sendText(text); }
catch (Exception e) {}
}
public void resetFromHint() {
if(!triviaOn) return;
hintThread.quit();
hintThread = null;
oldAnswer = answer;
try {Thread.sleep(5000);}
catch (Exception e2) {out.showMessage("Didn't sleep after reset");}
findQuestion();
try {Thread.sleep(5000);}
catch (Exception e2) {out.showMessage("Didn't sleep after reset");}
try {askQuestion(); }
catch (Exception e) {}
}
public void sayToConsole(String text) {
try{ out.showMessage(text);}
catch (Exception e) {}
}


public void emote(String user, String text, int ping, int flags) throws IOException, PluginException {}
public void whisperFrom(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void whisperTo(String user, String statstring, int ping, int flags) throws IOException, PluginException {}

public void userShow(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void userJoin(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void userLeave(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void userFlags(String user, String statstring, int ping, int flags) throws IOException, PluginException {}

public void error(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void info(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void broadcast(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
public void channel(String user, String statstring, int ping, int flags) throws IOException, PluginException {}
private String getSetting(String name) {
return out.getLocalSetting(getName(), name);
}

public void commandExecuted(String user, String command, String[] args, int loudness, Object data) throws PluginException, IOException {
if (command.equalsIgnoreCase("trivia")) {
if(triviaOn) {
triviaOn=false;
out.sendText("Trivia Turned Off");
questionAsked = false;
}
else {
triviaOn=true;
out.sendText("Trivia Turned On");
findQuestion();
askQuestion();
}
}
if (command.equalsIgnoreCase("top3")) {
//Get a vector of all players in class
Vector players = new Vector(scoreTable.keySet());
Vector scores = new Vector();
for(int j=0;j<players.size();j++) {
scores.add(scoreTable.get(players.elementAt(j)).getTotalCorrect());
}
Vector sortedScores = (Vector)scores.clone();
Collections.sort(sortedScores);
int sortedSize = sortedScores.size();
    for(int i=0;i<scores.size();i++) {
//if first place
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-1)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-2)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-3)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
    }
}
if (command.equalsIgnoreCase("streak")) {
//Get a vector of all players in class
Vector players = new Vector(scoreTable.keySet());
Vector scores = new Vector();
for(int j=0;j<players.size();j++) {
scores.add(scoreTable.get(players.elementAt(j)).getStreak());
}
Vector sortedScores = (Vector)scores.clone();
Collections.sort(sortedScores);
int sortedSize = sortedScores.size();
    for(int i=0;i<scores.size();i++) {
//if first place
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-1)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-2)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
if(scores.elementAt(i) == sortedScores.elementAt(sortedSize-3)) {
String player = (String)players.elementAt(i);
out.sendText(player + " " + scores.elementAt(i));
}
    }
}
}

private class HintThread extends Thread {
private int hintDelay,hintMax,answerLength;
private char hintChar;
private volatile Thread thisThread;
public HintThread(int hintDelay,int hintMax,int answerLength, char hintChar) {
setPriority(MIN_PRIORITY);
this.hintDelay = hintDelay;
this.hintMax = hintMax;
this.answerLength = answerLength;
this.hintChar = hintChar;
this.setName("Hint-thread");
}
public void quit() {
thisThread = null;
  }
public void run() {
thisThread = Thread.currentThread();
if(!triviaOn || !questionAsked) return; //Does this work in a thread?

// setPriority(MIN_PRIORITY);
int currentHint;
char[] hint = new char[answerLength];
for(int i=0;i<answerLength;i++) {
hint[i] = hintChar; //Make hint word all ----
}
try{
for(currentHint=0;currentHint<hintMax;currentHint++)
{
Thread.sleep(hintDelay);
//give hint - 1/4(?) of the letters - would be nice to make this configurable
//This is done by getting a random char of the string and replacing it with the correct one
for(int j=0;j<(answerLength/4);j++) {
hint[j+((answerLength/4)*currentHint)] = answer.charAt(j+((answerLength/4)*currentHint));
}
if(thisThread == null  || questionAsked == false || oldAnswer == answer) {out.showMessage("I should be returning"); return;}
else{
out.sendTextPriority("Hint: " + String.valueOf(hint),PRIORITY_NORMAL);
}
}
Thread.sleep(hintDelay*2);
if(thisThread == null  || questionAsked == false || oldAnswer == answer) {out.showMessage("I should be returning"); return;}
else {
out.sendTextPriority("The answer is: " + answer,10);
}
questionAsked=false;
out.showMessage("hintDelay is: " + hintDelay);
resetFromHint();
}catch(Exception e){}
}
}
public static class PlayerScore implements Serializable { //Class for keeping player scores

String playerName;
int longestStreak=0;
int totalCorrect=0;

public PlayerScore() {}
public PlayerScore(String name) {
playerName = name;
}
public void setTotalCorrect(int newCorrect) {
totalCorrect = newCorrect;
}
public void incTotalCorrect() {
totalCorrect++;
}
public void setLongestStreak(int streak) {
longestStreak=streak;
}
public void incLongestStreak() {
longestStreak++;
}
public String getName() {
return playerName;
}
public int getStreak() {
return longestStreak;
}
public int getTotalCorrect() {
return totalCorrect;
}
}
}



Offline Allanon

  • Newbie
  • *
  • Posts: 5
    • View Profile
    • Warcraft III : ecLipse Team
Re: Trivia Bot
« Reply #27 on: November 10, 2006, 09:55:38 am »
Code: [Select]
Loading plugin: file://home/korwin/javaop2/Plugins/WartTrivia.jar
java.lang.UnsupportedClassVersionError: PluginMain (Unsupported major.minor version 49.0)
What's wrong ?
[I wanna change this World, but God don't give me sources]

Offline Wart

  • Newbie
  • *
  • Posts: 17
    • View Profile
Re: Trivia Bot
« Reply #28 on: November 11, 2006, 01:11:05 am »
Code: [Select]
Loading plugin: file://home/korwin/javaop2/Plugins/WartTrivia.jar
java.lang.UnsupportedClassVersionError: PluginMain (Unsupported major.minor version 49.0)
What's wrong ?
Sounds like your version of java is a little out of date.. Be sure you have the newest :)

Offline Allanon

  • Newbie
  • *
  • Posts: 5
    • View Profile
    • Warcraft III : ecLipse Team
Re: Trivia Bot
« Reply #29 on: November 13, 2006, 05:12:53 am »
Thanks  ;)
[I wanna change this World, but God don't give me sources]