mirror of
https://github.com/raeleus/Hashtag-DnD.git
synced 2025-07-05 13:10:28 -04:00
385 lines
No EOL
21 KiB
JavaScript
385 lines
No EOL
21 KiB
JavaScript
const modifier = (text) => {
|
|
if (state.show == null) return { text }
|
|
|
|
var character = getCharacter()
|
|
var possessiveName = character == null ? null : getPossessiveName(character.name)
|
|
var type = history[history.length - 1].type
|
|
const originalText = text
|
|
text = type != "story" ? "" : history[history.length - 1].text.endsWith("\n") ? "" : "\n"
|
|
|
|
switch (state.show) {
|
|
case "create":
|
|
switch (state.createStep) {
|
|
case 0:
|
|
text += `***CHARACTER CREATION***\nCharacter: ${state.tempCharacter.name}\nWould you like to use a prefab character? (y/n/q to quit)\n`
|
|
break
|
|
case 1:
|
|
text += `What class is your character?\n`
|
|
break
|
|
case 2:
|
|
text += `You rolled the following stat dice: ${state.statDice}\nChoose your abilities in order from highest to lowest\n1. Strength: Physical power and endurance\n2. Dexterity: Agility and coordination\n3. Constitution: Toughness and physique \n4. Intelligence: Reasoning and memory\n5. Wisdom: Judgement and insight\n6. Charisma: Force of personality and persuasiveness\n\nEnter the numbers with spaces between or q to quit.\n`
|
|
break
|
|
case 3:
|
|
text += `What ability is your spell casting ability?\n1. Intelligence\n2. Wisdom\n3. Charisma\n4. Not a spell caster\nq to quit\n`
|
|
break
|
|
case 4:
|
|
text += `Enter a short summary about your character or q to quit\n`
|
|
break
|
|
case 100:
|
|
text += `What character will you choose?\n1. Fighter: A skilled melee warrior specializing in weapons and armor.\n2. Cleric: A follower of a deity that can call on divine power.\n3. Rogue: An expert in stealth, subterfuge, and exploitation.\n4. Ranger: A talented hunter adept in tracking, survival, and animal handling.\n5. Barbarian: Combat expert focused on brute strength and raw fury.\n6. Bard: A musician that can transform song and word into magic.\n7. Druid: Commands the natural world to cast spells and harness its power.\n8. Monk: A martial artist who has mastered melee and unarmed combat.\n9. Paladin: A virtuous holy warrior with expertise in armor and mysticism.\n10. Wizard: An expert in magic ability who found their power through arcane knowledge.\n11. Sorcerer: A masterful spellcaster deriving their power from an innate source.\n12. Warlock: A magic user granted ability by a pact with a powerful patron.\n13. Artificer: An inventor and alchemist capable of imbuing objects with magic.\n\nEnter the number or q to quit.\n`
|
|
break
|
|
case 500:
|
|
text += `${state.tempCharacter.name} the ${state.tempCharacter.className} has been created.\nType #bio to see a summary of your character.\n***********\n`
|
|
break;
|
|
case null:
|
|
text += `[Character creation has been aborted!]\n`
|
|
break
|
|
}
|
|
break
|
|
case "bio":
|
|
text += `*** ${possessiveName.toUpperCase()} BIO ***\n`
|
|
text += `Class: ${character.className}\n`
|
|
text += `Health: ${character.health}/${getHealthMax()}\n`
|
|
text += `Experience: ${character.experience}\n`
|
|
text += `Level: ${getLevel(character.experience)}\n`
|
|
var nextLevel = getNextLevelXp(character.experience)
|
|
text += `Next level at: ${nextLevel == - 1 ? "(at maximum)": nextLevel + " xp"}\n\n`
|
|
text += `-ABILITIES-\n`
|
|
|
|
character.stats.forEach(function(x) {
|
|
text += `* ${toTitleCase(x.name)} ${x.value}\n`
|
|
})
|
|
|
|
text += `----\n\n`
|
|
|
|
text += `-SKILLS-\n`
|
|
|
|
character.skills.forEach(function(x) {
|
|
const stat = character.stats.find(y => y.name.toLowerCase() == x.stat.toLowerCase())
|
|
|
|
var statModifier = stat != null ? getModifier(stat.value): 0
|
|
var totalModifier = x.modifier + statModifier
|
|
var modifier = x.modifier
|
|
|
|
if (statModifier >= 0) statModifier = `+${statModifier}`
|
|
if (totalModifier >= 0) totalModifier = `+${totalModifier}`
|
|
if (modifier >= 0) modifier = `+${modifier}`
|
|
|
|
text += `* ${toTitleCase(x.name)} ${totalModifier} = ${toTitleCase(x.stat)} ${statModifier} Proficiency ${modifier}\n`
|
|
})
|
|
|
|
text += `----\n\n`
|
|
|
|
text += `Melee Ability: ${character.meleeStat == null ? "none" : character.meleeStat}\n\n`
|
|
text += `Ranged Ability: ${character.rangedStat == null ? "none" : character.rangedStat}\n\n`
|
|
text += `Spellcasting Ability: ${character.spellStat == null ? "none" : character.spellStat}\n\n`
|
|
|
|
if (character.spellStat != null) {
|
|
text += `-SPELLS-\n`
|
|
|
|
character.spells.forEach(function(x) {
|
|
text += `* ${toTitleCase(x)}\n`
|
|
})
|
|
|
|
text += `----\n\n`
|
|
}
|
|
|
|
text += `-INVENTORY-\n`
|
|
|
|
character.inventory.forEach(function(x) {
|
|
text += `* ${x.quantity} ${toTitleCase(x.name.plural(x.quantity == 1))}\n`
|
|
})
|
|
|
|
text += `----\n\n`
|
|
|
|
text += `Summary: ${character.summary}\n\n`
|
|
|
|
text += `**************\n\n`
|
|
break
|
|
case "showNotes":
|
|
text += "*** NOTES ***"
|
|
var counter = 1
|
|
state.notes.forEach(function(x) {
|
|
text += `\n${counter++}. ${x}`
|
|
})
|
|
if (state.notes.length == 0) text += "\nThere are no notes!"
|
|
text += "\n**************\n\n"
|
|
break
|
|
case "clearNotes":
|
|
text += "[Notes cleared successfully]\n"
|
|
break
|
|
case "inventory":
|
|
text += `*** ${possessiveName.toUpperCase()} INVENTORY ***`
|
|
if (character.inventory.length > 0) {
|
|
character.inventory.forEach(function(x) {
|
|
text += `\n* ${x.quantity} ${toTitleCase(x.name.plural(x.quantity == 1))}`
|
|
})
|
|
} else {
|
|
text += `\n${possessiveName} inventory is empty!`
|
|
}
|
|
text += "\n******************\n\n"
|
|
break
|
|
case "characters":
|
|
text += `*** CHARACTERS ***`
|
|
if (state.characters.length > 0) {
|
|
state.characters.forEach(function(x) {
|
|
text += `\n* ${toTitleCase(x.name)} the ${toTitleCase(x.className)}: ${x.summary}`
|
|
})
|
|
} else {
|
|
text += `\n${possessiveName} inventory is empty!`
|
|
}
|
|
text += "\n******************\n\n"
|
|
break
|
|
case "spellbook":
|
|
text += `*** ${possessiveName.toUpperCase()} SPELLBOOK ***`
|
|
if (character.spells.length > 0) {
|
|
character.spells.forEach(function(x) {
|
|
text += "\n* " + toTitleCase(x)
|
|
})
|
|
} else {
|
|
text += `\n${possessiveName} spellbook is empty!`
|
|
}
|
|
text += "\n******************\n\n"
|
|
break
|
|
case "stats":
|
|
text += `*** ${possessiveName.toUpperCase()} ABILITIES ***\n`
|
|
if (character.stats.length > 0) {
|
|
character.stats.forEach(function(x) {
|
|
text += `* ${toTitleCase(x.name)} ${x.value}\n`
|
|
})
|
|
} else {
|
|
text += `${character.name} has no abilities!\n`
|
|
}
|
|
text += "******************\n\n"
|
|
break
|
|
case "skills":
|
|
text += `*** ${possessiveName.toUpperCase()} SKILLS ***\n`
|
|
if (character.skills.length > 0) {
|
|
character.skills.forEach(function(x) {
|
|
const stat = character.stats.find(y => y.name.toLowerCase() == x.stat.toLowerCase())
|
|
|
|
var statModifier = stat != null ? getModifier(stat.value): 0
|
|
var totalModifier = x.modifier + statModifier
|
|
var modifier = x.modifier
|
|
|
|
if (statModifier >= 0) statModifier = `+${statModifier}`
|
|
if (totalModifier >= 0) totalModifier = `+${totalModifier}`
|
|
if (modifier >= 0) modifier = `+${modifier}`
|
|
|
|
text += `* ${toTitleCase(x.name)} ${totalModifier} = ${toTitleCase(x.stat)} ${statModifier} Proficiency ${modifier}\n`
|
|
})
|
|
} else {
|
|
text += `${character.name} has no skills!\n`
|
|
}
|
|
text += "******************\n\n"
|
|
break
|
|
case "locations":
|
|
text += `Player location: ${state.location == null ? "" : state.location + " "}(${state.x},${state.y})\n`
|
|
text += `*** LOCATIONS ***\n`
|
|
if (state.locations.length > 0) {
|
|
var index = 0
|
|
state.locations.forEach(function(location) {
|
|
var distance = pointDistance(state.x, state.y, location.x, location.y).toFixed(1)
|
|
text += `${++index}. ${toTitleCase(location.name)} (${location.x},${location.y}) Distance: ${distance}\n`
|
|
})
|
|
} else {
|
|
text += `No locations have been discovered!\n`
|
|
}
|
|
text += "******************\n\n"
|
|
break
|
|
case "map":
|
|
text += `A 11x11 map of the area surrounding (${state.x},${state.y}):\n`
|
|
var map = mapGenerate()
|
|
map = mapReplace(map, state.x, state.y, "@")
|
|
state.locations.forEach(location => {
|
|
map = mapReplace(map, location.x, location.y, location.name.substring(0, 1).toUpperCase())
|
|
})
|
|
text += map
|
|
break
|
|
case "none":
|
|
text += " "
|
|
break
|
|
case "prefix":
|
|
text = state.prefix + originalText
|
|
break
|
|
case "clearInventory":
|
|
text += `[${possessiveName} inventory has been emptied]\n`
|
|
break
|
|
case "reset":
|
|
text += "[All settings have been reset]\n"
|
|
break
|
|
case "help":
|
|
|
|
text += "--Basic Hashtags--"
|
|
text += "\n#roll (advantage|disadvantage) (dice_value)"
|
|
text += "\n Rolls a die/dice and shows the result. dice_value can be in the following formats 5d20 or d20 or 20. The parameters can be listed in any order."
|
|
text += "\n#generatename (male|female) (fantasy|modern|scifi|nordic)"
|
|
text += "\n Retrieves a random name from a list of names in the specified gender and genre. The parameters can be listed in any order."
|
|
text += "\n#shownotes"
|
|
text += "\n Shows all the notes."
|
|
text += "\n#note message"
|
|
text += "\n Adds the specified message as a note. Quotes are not necessary."
|
|
text += "\n#clearnotes"
|
|
text += "\n Removes all notes."
|
|
text += "\n#removenote value"
|
|
text += "\n Removes the specified note as indicated by the number listed in #shownotes. To delete multiple notes, type the numbers with spaces or commas between them. This is safer than calling #removenote multiple times because the numbers shift as notes are deleted."
|
|
|
|
text += "\n\n--Characters--"
|
|
text += "\n#setup"
|
|
text += "\n Launches the create character setup."
|
|
text += "\n#bio"
|
|
text += "\n Shows the character's abilities, skills, spells, inventory, and everything else about this character."
|
|
text += "\n#renamecharacter new_name"
|
|
text += "\n Renames the character to the new name. All abilities, skills, inventory, etc. will remain the same. Quotes are not necessary."
|
|
text += "\n#clonecharacter new_name"
|
|
text += "\n Copies the abilities, skills, inventory, etc. of the current character to a new character with the name new_name. Quotes are not necessary."
|
|
text += "\n#setclass class"
|
|
text += "\n Sets the class of the character for player reference. Quotes are not necessary."
|
|
text += "\n#setsummary summary"
|
|
text += "\n Sets the summary of the character for player reference. Quotes are not necessary."
|
|
text += "\n#sethealth value"
|
|
text += "\n Sets the character's health to specified value. It's capped at the character's max health."
|
|
text += "\n#heal value"
|
|
text += "\n Increases the character's health by the specified value. It's capped at the character's max health."
|
|
text += "\n#damage value"
|
|
text += "\n Decreases the character's health by the specified value. Reaching 0 causes the character to become \"unconcious\"."
|
|
text += "\n#rest"
|
|
text += "\n Sets all of the characters' health to their maximums. Use #shortrest to only restore half health."
|
|
text += "\n#setxp value"
|
|
text += "\n Sets the character's experience to the specified value."
|
|
text += "\n#addxp value"
|
|
text += "\n Increases the character's experience by the specified value. The player is notified if there is a level up."
|
|
text += "\n#setautoxp value"
|
|
text += "\n Automatically increases the experience of all party members when a #try, #attack, or #cast is called. The amount of experience is scaled based on the difficulty class of the check with any check 20 or higher will result in the maximum specified by value. Set to 0 to disable."
|
|
text += "\n#showautoxp"
|
|
text += "\n Shows the value of the auto xp."
|
|
text += "\n#levelup"
|
|
text += "\n Increases the character's experience by the exact amount needed to reach the next level."
|
|
text += "\n#setdefaultdifficulty (difficulty_class or effortless|easy|medium|hard|impossible)"
|
|
text += "\n Sets the default difficulty for #check, #try, #attack, and #cast when a difficulty is not specified. The normal default is 10 (easy). If you do not pass any parameters, the default will be set to 10 (easy)."
|
|
text += "\n#showdefaultdifficulty"
|
|
text += "\n Shows the default difficulty for #check, #try, #attack, and #cast when a difficulty is not specified. The normal default is 10 (easy)."
|
|
text += "\n#showcharacters"
|
|
text += "\n Lists all current characters and their classes/summaries."
|
|
text += "\n#removecharacter name"
|
|
text += "\n Removes the character that has the indicated name."
|
|
|
|
text += "\n\n--Character Checks--"
|
|
text += "\n#check (ability|skill) (advantage|disadvantage) (difficulty_class or effortless|easy|medium|hard|impossible)"
|
|
text += "\n Rolls a d20 and compares the result (modified by the character's ability/skill) to the specified difficulty. The parameters can be listed in any order."
|
|
text += "\n#try (ability|skill) (advantage|disadvantage) (difficulty_class or effortless|easy|medium|hard|impossible) task"
|
|
text += "\n Attempts to do the task based on the character's ability/skill against the specified difficulty. Quotes are not necessary."
|
|
text += "\n#attack (ranged) (advantage|disadvantage) (ac or effortless|easy|medium|hard|impossible) target"
|
|
text += "\n Attacks the specified target with a melee (the default) or ranged attack. The roll is compared against the specified AC which will determine if the attack succeeds or misses. The parameters can be listed in any order, except the target must be listed last. Quotes are not necessary."
|
|
text += "\n#cast (advantage|disadvantage) (difficulty_class or effortless|easy|medium|hard|impossible) spell(target)"
|
|
text += "\n Character will cast the indicated spell if the spell is in their spellbook. It will be a targeted spell if a target is indicated. The roll is modified by the spell casting ability of the character. You may type a phrase without quotes for spell such as \"cast fire bolt at the giant chicken\". The parameters can be listed in any order, except the target must be listed last. Quotes are not necessary."
|
|
|
|
text += "\n\n--Abilities--"
|
|
text += "\n#setability ability value"
|
|
text += "\n Adds the ability to the character if necessary and sets it to the specified value. Quotes are required for abilities with spaces."
|
|
text += "\n#showabilities"
|
|
text += "\n Shows the character's list of abilities."
|
|
text += "\n#removeability ability"
|
|
text += "\n Removes the ability from the character's list of abilities. Quotes are not necessary."
|
|
text += "\n#clearabilities"
|
|
text += "\n Removes all abilities from the character."
|
|
text += "\n#setspellability ability"
|
|
text += "\n Sets the ability that affects the modifier for #cast. Quotes are not necessary."
|
|
text += "\n#setmeleeability ability"
|
|
text += "\n Sets the character's ability modifier that affects melee attacks. Quotes are not necessary."
|
|
text += "\n#setrangedability ability"
|
|
text += "\n Sets the character's ability modifier that affects ranged attacks. Quotes are not necessary."
|
|
|
|
text += "\n\n--Skills--"
|
|
text += "\n#setskill skill (ability) value"
|
|
text += "\n Adds the skill to the character if necessary, and associates it with the specified ability and value. The ability is optional only if this is an existing skill. New skills need an ability specified."
|
|
text += "\n#showskills"
|
|
text += "\n Shows the character's list of skills"
|
|
text += "\n#removeskill"
|
|
text += "\n Removes the skill from the character's list of skills."
|
|
text += "\n#clearskills"
|
|
text += "\n Removes all skills from the character."
|
|
|
|
text += "\n\n--Inventory--"
|
|
text += "\n#take (quantity) item"
|
|
text += "\n Adds the specified quantity of item to the character's inventory. If a quantity is omitted, it's assumed to be 1. The words the, a, and an are ignored. Quotes are not necessary."
|
|
text += "\n#buy (buy_quantity) buy_item (sell_quantity) sell_item"
|
|
text += "\n Adds the specified buy_quantity of the buy_item to the character's inventory and also removes the sell_quantity of sell_item. If quantities are omitted, they are assumed to be 1. Quotes are necessary for items with spaces in the name. The words for, with, the, a, and an are ignored."
|
|
text += "\n#sell (sell_quantity) sell_item (buy_quantity) buy_item"
|
|
text += "\n Just like #buy, but with the parameters reversed. Adds the specified buy_quantity of the buy_item to the character's inventory and also removes the sell_quantity of sell_item. If quantities are omitted, they are assumed to be 1. The words for, with, the, a, and an are ignored. Quotes are necessary for items with spaces in the name."
|
|
text += "\n#drop (quantity or all|every) item"
|
|
text += "\n Removes the specified quantity of item from the character's inventory. If a quantity is omitted, it's assumed to be 1. The words the, a, and an are ignored. Quotes are not necessary."
|
|
text += "\n#give other_character (quantity or all|every) item"
|
|
text += "\n Removes the quantity of item from the character's inventory and adds it to the other_character's inventory. If a quantity is omitted, it's assumed to be 1. The words the, a, and an are ignored. Quotes are not necessary for the item."
|
|
text += "\n#renameitem original_name new_name"
|
|
text += "\n Renames the item indicated by original_name to the new_name. The quantity remains the same. Quotes are necessary for items with spaces in the name."
|
|
text += "\n#inventory"
|
|
text += "\n Shows the items in the inventory of the character."
|
|
text += "\n#clearinventory"
|
|
text += "\n Removes all items from the character's inventory."
|
|
|
|
text += "\n\n--Spells--"
|
|
text += "\n#learnspell spell"
|
|
text += "\n Adds the specified spell to the character's spellbook. Creates a story card if necessary. Quotes are not necessary."
|
|
text += "\n#forgetSpell spell"
|
|
text += "\n Removes the specified spell from the character's spellbook. Quotes are not necessary."
|
|
text += "\n#clearspells"
|
|
text += "\n Removes all spells from the character's spellbook."
|
|
text += "\n#spellbook"
|
|
text += "\n Shows the list of spells that the character has learned."
|
|
|
|
text += "\n\n--Locations--"
|
|
text += "\n#createlocation (x) (y) location_name"
|
|
text += "\n Creates a location at the given coordinates. If the coordinates are not provided, they are randomized within a range of 10 units from the player's current location. Multiple locations may exist at the same coordinates. A story card is created for the location. Quotes are not necessary."
|
|
text += "\n#goto (x) (y) or (location_name)"
|
|
text += "\n Makes the characters travel to the location specified by the coordinates or location_name. You must provide at least one or the other. If the location does not exist, it is created at your current coordinates. If you only specify coordinates, you will go to the first location at those coordinates. Quotes are not necessary."
|
|
text += "\n#getlocation"
|
|
text += "\n Returns the coordinates that the player's party is at. It will also list a location if a location was specified when using #goto."
|
|
text += "\n#showlocations"
|
|
text += "\n Shows a list of all discovered locations with their coordinates and their distance from the player's current location."
|
|
text += "\n#removelocation location_name or location_number"
|
|
text += "\n Removes the specified location by location_name or location_number as listed in #showlocations. To delete multiple locations, type the numbers with spaces or commas between them. This is safer than calling #removenote multiple times because the numbers shift as notes are deleted. Quotes are not necessary."
|
|
text += "\n#clearlocations"
|
|
text += "\n Deletes all discovered locations."
|
|
text += "\n#map"
|
|
text += "\n Generates an 11x11 ASCII map of the surrounding locations centered at the player location."
|
|
|
|
text += "\n\n--Danger Zone--"
|
|
text += "\n#reset"
|
|
text += "\n Removes all characters and changes all settings to their defaults. Use with caution!"
|
|
|
|
text += "\n\n--Other--"
|
|
text += "\n#help"
|
|
text += "\n This long ass help menu. I am paid by lines of codes."
|
|
|
|
break
|
|
}
|
|
|
|
state.show = null
|
|
return { text }
|
|
}
|
|
|
|
const mapLineBreak = "- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -\n"
|
|
const mapLine = "| * * * * * * * * * * * |\n"
|
|
|
|
function mapGenerate() {
|
|
var map = mapLineBreak
|
|
for (var i = 0; i < 11; i++) map += mapLine
|
|
map += mapLineBreak
|
|
return map
|
|
}
|
|
|
|
function mapReplace(map, x, y, character) {
|
|
x += 5 - state.x
|
|
y += 5 - state.y
|
|
|
|
if (x < 0 || x > 10 || y < 0 || y > 10) return map
|
|
|
|
index = mapLineBreak.length + 6 + x * 6 + y * mapLine.length
|
|
|
|
map = map.replaceAt(index, character)
|
|
|
|
return map
|
|
}
|
|
|
|
modifier(text) |