2019-04-11 10:37:40 +02:00
|
|
|
import { Engine } from "./engine";
|
|
|
|
import { Player } from "./Player";
|
|
|
|
import { Settings } from "./Settings/Settings";
|
|
|
|
|
|
|
|
import { initializeMainMenuLinks } from "./ui/MainMenu/Links";
|
2021-03-14 07:08:24 +01:00
|
|
|
import { LiteratureNames } from "./Literature/data/LiteratureNames";
|
2019-04-11 10:37:40 +02:00
|
|
|
|
|
|
|
import { exceptionAlert } from "../utils/helpers/exceptionAlert";
|
|
|
|
import { clearEventListeners } from "../utils/uiHelpers/clearEventListeners";
|
|
|
|
import { createElement } from "../utils/uiHelpers/createElement";
|
|
|
|
import { createPopup } from "../utils/uiHelpers/createPopup";
|
|
|
|
import { removeElementById } from "../utils/uiHelpers/removeElementById";
|
|
|
|
|
2017-08-30 19:44:29 +02:00
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Ordered array of keys to Interactive Tutorial Steps
|
2018-08-29 21:06:21 +02:00
|
|
|
const orderedITutorialSteps = [
|
|
|
|
"Start",
|
2019-04-13 03:22:46 +02:00
|
|
|
"GoToCharacterPage", // Click on 'Stats' page
|
|
|
|
"CharacterPage", // Introduction to 'Stats' page
|
|
|
|
"CharacterGoToTerminalPage", // Go back to Terminal
|
|
|
|
"TerminalIntro", // Introduction to Terminal
|
|
|
|
"TerminalHelp", // Using 'help' Terminal command
|
|
|
|
"TerminalLs", // Using 'ls' Terminal command
|
|
|
|
"TerminalScan", // Using 'scan' Terminal command
|
|
|
|
"TerminalScanAnalyze1", // Using 'scan-analyze' Terminal command
|
|
|
|
"TerminalScanAnalyze2", // Using 'scan-analyze 3' Terminal command
|
|
|
|
"TerminalConnect", // Connecting to foodnstuff
|
|
|
|
"TerminalAnalyze", // Analyzing foodnstuff
|
|
|
|
"TerminalNuke", // NUKE foodnstuff
|
|
|
|
"TerminalManualHack", // Hack foodnstuff
|
|
|
|
"TerminalHackingMechanics", // Explanation of hacking mechanics
|
|
|
|
"TerminalCreateScript", // Create a script using 'nano'
|
|
|
|
"TerminalTypeScript", // Script Editor page - Type script and then save & close
|
|
|
|
"TerminalFree", // Using 'Free' Terminal command
|
|
|
|
"TerminalRunScript", // Running script using 'run' Terminal command
|
2018-08-29 21:06:21 +02:00
|
|
|
"TerminalGoToActiveScriptsPage",
|
|
|
|
"ActiveScriptsPage",
|
|
|
|
"ActiveScriptsToTerminal",
|
|
|
|
"TerminalTailScript",
|
|
|
|
"GoToHacknetNodesPage",
|
|
|
|
"HacknetNodesIntroduction",
|
|
|
|
"HacknetNodesGoToWorldPage",
|
|
|
|
"WorldDescription",
|
|
|
|
"TutorialPageInfo",
|
2021-04-30 05:52:56 +02:00
|
|
|
"End",
|
2018-08-29 21:06:21 +02:00
|
|
|
]
|
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Create an 'enum' for the Steps
|
2018-08-29 21:06:21 +02:00
|
|
|
const iTutorialSteps = {};
|
|
|
|
for (let i = 0; i < orderedITutorialSteps.length; ++i) {
|
|
|
|
iTutorialSteps[orderedITutorialSteps[i]] = i;
|
2017-05-05 23:27:35 +02:00
|
|
|
}
|
|
|
|
|
2018-08-29 21:06:21 +02:00
|
|
|
var ITutorial = {
|
2019-04-13 03:22:46 +02:00
|
|
|
currStep: 0, // iTutorialSteps.Start
|
2018-08-29 21:06:21 +02:00
|
|
|
isRunning: false,
|
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Keeps track of whether each step has been done
|
2018-08-29 21:06:21 +02:00
|
|
|
stepIsDone: {},
|
|
|
|
}
|
2017-05-05 23:27:35 +02:00
|
|
|
|
|
|
|
function iTutorialStart() {
|
2019-04-13 03:22:46 +02:00
|
|
|
// Initialize Interactive Tutorial state by settings 'done' for each state to false
|
2018-08-29 21:06:21 +02:00
|
|
|
ITutorial.stepIsDone = {};
|
|
|
|
for (let i = 0; i < orderedITutorialSteps.length; ++i) {
|
|
|
|
ITutorial.stepIsDone[i] = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Engine.loadTerminalContent();
|
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Don't autosave during this interactive tutorial
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.Counters.autoSaveCounter = Infinity;
|
|
|
|
ITutorial.currStep = 0;
|
|
|
|
ITutorial.isRunning = true;
|
2017-08-15 22:22:46 +02:00
|
|
|
|
2017-05-05 23:27:35 +02:00
|
|
|
document.getElementById("interactive-tutorial-container").style.display = "block";
|
2017-08-15 22:22:46 +02:00
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Exit tutorial button
|
2017-05-05 23:27:35 +02:00
|
|
|
var exitButton = clearEventListeners("interactive-tutorial-exit");
|
|
|
|
exitButton.addEventListener("click", function() {
|
|
|
|
iTutorialEnd();
|
2017-05-06 08:24:01 +02:00
|
|
|
return false;
|
2017-05-05 23:27:35 +02:00
|
|
|
});
|
2017-08-15 22:22:46 +02:00
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Back button
|
2017-05-15 07:09:14 +02:00
|
|
|
var backButton = clearEventListeners("interactive-tutorial-back");
|
|
|
|
backButton.addEventListener("click", function() {
|
|
|
|
iTutorialPrevStep();
|
|
|
|
return false;
|
|
|
|
});
|
2018-08-29 21:06:21 +02:00
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Next button
|
2018-08-29 21:06:21 +02:00
|
|
|
var nextButton = clearEventListeners("interactive-tutorial-next");
|
|
|
|
nextButton.addEventListener("click", function() {
|
|
|
|
iTutorialNextStep();
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
|
|
|
|
iTutorialEvaluateStep();
|
2017-05-05 23:27:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function iTutorialEvaluateStep() {
|
2021-03-08 04:46:50 +01:00
|
|
|
if (!ITutorial.isRunning) {return;}
|
2018-08-29 21:06:21 +02:00
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Disable and clear main menu
|
2018-08-29 21:06:21 +02:00
|
|
|
var terminalMainMenu = clearEventListeners("terminal-menu-link");
|
|
|
|
var statsMainMenu = clearEventListeners("stats-menu-link");
|
|
|
|
var activeScriptsMainMenu = clearEventListeners("active-scripts-menu-link");
|
|
|
|
var hacknetMainMenu = clearEventListeners("hacknet-nodes-menu-link");
|
|
|
|
var cityMainMenu = clearEventListeners("city-menu-link");
|
|
|
|
var tutorialMainMenu = clearEventListeners("tutorial-menu-link");
|
|
|
|
terminalMainMenu.removeAttribute("class");
|
|
|
|
statsMainMenu.removeAttribute("class");
|
|
|
|
activeScriptsMainMenu.removeAttribute("class");
|
|
|
|
hacknetMainMenu.removeAttribute("class");
|
|
|
|
cityMainMenu.removeAttribute("class");
|
|
|
|
tutorialMainMenu.removeAttribute("class");
|
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Interactive Tutorial Next button
|
2018-08-29 21:06:21 +02:00
|
|
|
var nextBtn = document.getElementById("interactive-tutorial-next");
|
|
|
|
|
|
|
|
switch(ITutorial.currStep) {
|
2017-05-05 23:27:35 +02:00
|
|
|
case iTutorialSteps.Start:
|
2017-05-15 07:09:14 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-05-06 08:24:01 +02:00
|
|
|
iTutorialSetText("Welcome to Bitburner, a cyberpunk-themed incremental RPG! " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"The game takes place in a dark, dystopian future...The year is 2077...<br><br>" +
|
2017-09-05 03:39:00 +02:00
|
|
|
"This tutorial will show you the basics of the game. " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"You may skip the tutorial at any time.");
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "inline-block";
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.GoToCharacterPage:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("Let's start by heading to the Stats page. Click the 'Stats' tab on " +
|
2017-05-05 23:27:35 +02:00
|
|
|
"the main navigation menu (left-hand side of the screen)");
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "none";
|
2017-08-15 22:22:46 +02:00
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Flash 'Stats' menu and set its tutorial click handler
|
2018-08-29 21:06:21 +02:00
|
|
|
statsMainMenu.setAttribute("class", "flashing-button");
|
|
|
|
statsMainMenu.addEventListener("click", function() {
|
2017-05-05 23:27:35 +02:00
|
|
|
Engine.loadCharacterContent();
|
|
|
|
iTutorialNextStep(); //Opening the character page will go to the next step
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case iTutorialSteps.CharacterPage:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadCharacterContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("The Stats page shows a lot of important information about your progress, " +
|
2017-07-22 00:54:55 +02:00
|
|
|
"such as your skills, money, and bonuses/multipliers. ")
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "inline-block";
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.CharacterGoToTerminalPage:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadCharacterContent();
|
2017-05-05 23:27:35 +02:00
|
|
|
iTutorialSetText("Let's head to your computer's terminal by clicking the 'Terminal' tab on the " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"main navigation menu.");
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "none";
|
2017-08-15 22:22:46 +02:00
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Flash 'Terminal' menu and set its tutorial click handler
|
2018-08-29 21:06:21 +02:00
|
|
|
terminalMainMenu.setAttribute("class", "flashing-button");
|
|
|
|
terminalMainMenu.addEventListener("click", function() {
|
2017-05-05 23:27:35 +02:00
|
|
|
Engine.loadTerminalContent();
|
|
|
|
iTutorialNextStep();
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalIntro:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-05-06 08:24:01 +02:00
|
|
|
iTutorialSetText("The Terminal is used to interface with your home computer as well as " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"all of the other machines around the world.");
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "inline-block";
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalHelp:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("Let's try it out. Start by entering the 'help' command into the Terminal " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"(Don't forget to press Enter after typing the command)");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal command
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalLs:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-09-05 03:39:00 +02:00
|
|
|
iTutorialSetText("The 'help' command displays a list of all available Terminal commands, how to use them, " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"and a description of what they do. <br><br>Let's try another command. Enter the 'ls' command");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal command
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalScan:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-05-06 08:24:01 +02:00
|
|
|
iTutorialSetText("'ls' is a basic command that shows all of the contents (programs/scripts) " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"on the computer. Right now, it shows that you have a program called 'NUKE.exe' on your computer. " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"We'll get to what this does later. <br><br>Using your home computer's terminal, you can connect " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"to other machines throughout the world. Let's do that now by first entering " +
|
2018-08-29 21:06:21 +02:00
|
|
|
"the 'scan' command.");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal command
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
2017-05-25 01:23:28 +02:00
|
|
|
case iTutorialSteps.TerminalScanAnalyze1:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-05-17 07:31:42 +02:00
|
|
|
iTutorialSetText("The 'scan' command shows all available network connections. In other words, " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"it displays a list of all servers that can be connected to from your " +
|
2017-05-25 01:23:28 +02:00
|
|
|
"current machine. A server is identified by either its IP or its hostname. <br><br> " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"That's great and all, but there's so many servers. Which one should you go to? " +
|
|
|
|
"The 'scan-analyze' command gives some more detailed information about servers on the " +
|
2017-05-25 01:23:28 +02:00
|
|
|
"network. Try it now");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal command
|
2017-05-25 01:23:28 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalScanAnalyze2:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("You just ran 'scan-analyze' with a depth of one. This command shows more detailed " +
|
|
|
|
"information about each server that you can connect to (servers that are a distance of " +
|
2017-05-25 01:23:28 +02:00
|
|
|
"one node away). <br><br> It is also possible to run 'scan-analyze' with " +
|
|
|
|
"a higher depth. Let's try a depth of two with the following command: 'scan-analyze 2'.")
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal command
|
2017-05-25 01:23:28 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalConnect:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-05-25 01:23:28 +02:00
|
|
|
iTutorialSetText("Now you can see information about all servers that are up to two nodes away, as well " +
|
2017-05-25 17:55:25 +02:00
|
|
|
"as figure out how to navigate to those servers through the network. You can only connect to " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"a server that is one node away. To connect to a machine, use the 'connect [ip/hostname]' command. You can type in " +
|
|
|
|
"the ip or the hostname, but dont use both.<br><br>" +
|
2017-05-25 01:23:28 +02:00
|
|
|
"From the results of the 'scan-analyze' command, we can see that the 'foodnstuff' server is " +
|
|
|
|
"only one node away. Let's connect so it now using: 'connect foodnstuff'");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal command
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalAnalyze:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("You are now connected to another machine! What can you do now? You can hack it!<br><br> In the year 2077, currency has " +
|
|
|
|
"become digital and decentralized. People and corporations store their money " +
|
|
|
|
"on servers and computers. Using your hacking abilities, you can hack servers " +
|
|
|
|
"to steal money and gain experience. <br><br> " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"Before you try to hack a server, you should run diagnostics using the 'analyze' command");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal command
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalNuke:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("When the 'analyze' command finishes running it will show useful information " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"about hacking the server. <br><br> For this server, the required hacking skill is only 1, " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"which means you can hack it right now. However, in order to hack a server " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"you must first gain root access. The 'NUKE.exe' program that we saw earlier on your " +
|
|
|
|
"home computer is a virus that will grant you root access to a machine if there are enough " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"open ports.<br><br> The 'analyze' results shows that there do not need to be any open ports " +
|
|
|
|
"on this machine for the NUKE virus to work, so go ahead and run the virus using the " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"'run NUKE.exe' command.");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal command
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalManualHack:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("You now have root access! You can hack the server using the 'hack' command. " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"Try doing that now.");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal command
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
2017-05-17 07:31:42 +02:00
|
|
|
case iTutorialSteps.TerminalHackingMechanics:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("You are now attempting to hack the server. Note that performing a hack takes time and " +
|
|
|
|
"only has a certain percentage chance " +
|
|
|
|
"of success. This time and success chance is determined by a variety of factors, including " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"your hacking skill and the server's security level.<br><br>" +
|
2017-05-17 07:31:42 +02:00
|
|
|
"If your attempt to hack the server is successful, you will steal a certain percentage " +
|
|
|
|
"of the server's total money. This percentage is affected by your hacking skill and " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"the server's security level.<br><br>The amount of money on a server is not limitless. So, if " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"you constantly hack a server and deplete its money, then you will encounter " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"diminishing returns in your hacking.");
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "inline-block";
|
2017-05-17 07:31:42 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalCreateScript:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("Hacking is the core mechanic of the game and is necessary for progressing. However, " +
|
|
|
|
"you don't want to be hacking manually the entire time. You can automate your hacking " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"by writing scripts!<br><br>To create a new script or edit an existing one, you can use the 'nano' " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"command. Scripts must end with the '.script' extension. Let's make a script now by " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"entering 'nano foodnstuff.script' after the hack command finishes running (Sidenote: Pressing ctrl + c" +
|
2017-05-06 08:24:01 +02:00
|
|
|
" will end a command like hack early)");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal command
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalTypeScript:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadScriptEditorContent("foodnstuff.script", "");
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("This is the script editor. You can use it to program your scripts. Scripts are " +
|
|
|
|
"written in the Netscript language, a programming language created for " +
|
2017-05-17 07:31:42 +02:00
|
|
|
"this game. <strong style='background-color:#444;'>There are details about the Netscript language in the documentation, which " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"can be accessed in the 'Tutorial' tab on the main navigation menu. I highly suggest you check " +
|
2018-08-30 22:27:47 +02:00
|
|
|
"it out after this tutorial.</strong> For now, just copy " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"and paste the following code into the script editor: <br><br>" +
|
2017-08-15 22:22:46 +02:00
|
|
|
"while(true) { <br>" +
|
|
|
|
" hack('foodnstuff'); <br>" +
|
2017-05-06 08:24:01 +02:00
|
|
|
"}<br><br> " +
|
|
|
|
"For anyone with basic programming experience, this code should be straightforward. " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"This script will continuously hack the 'foodnstuff' server.<br><br>" +
|
2017-05-25 01:23:28 +02:00
|
|
|
"To save and close the script editor, press the button in the bottom left, or press ctrl + b.");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered in saveAndCloseScriptEditor() (Script.js)
|
2017-05-06 08:24:01 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalFree:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("Now we'll run the script. Scripts require a certain amount of RAM to run, and can be " +
|
|
|
|
"run on any machine which you have root access to. Different servers have different " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"amounts of RAM. You can also purchase more RAM for your home server.<br><br>To check how much " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"RAM is available on this machine, enter the 'free' command.");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal commmand
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalRunScript:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2018-01-20 05:47:57 +01:00
|
|
|
iTutorialSetText("We have 16GB of free RAM on this machine, which is enough to run our " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"script. Let's run our script using 'run foodnstuff.script'.");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal commmand
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalGoToActiveScriptsPage:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("Your script is now running! The script might take a few seconds to 'fully start up'. " +
|
|
|
|
"Your scripts will continuously run in the background and will automatically stop if " +
|
|
|
|
"the code ever completes (the 'foodnstuff.script' will never complete because it " +
|
|
|
|
"runs an infinite loop). <br><br>These scripts can passively earn you income and hacking experience. " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"Your scripts will also earn money and experience while you are offline, although at a " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"much slower rate. <br><br> " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"Let's check out some statistics for our running scripts by clicking the " +
|
|
|
|
"'Active Scripts' link in the main navigation menu.");
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "none";
|
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Flash 'Active Scripts' menu and set its tutorial click handler
|
2018-08-29 21:06:21 +02:00
|
|
|
activeScriptsMainMenu.setAttribute("class", "flashing-button");
|
|
|
|
activeScriptsMainMenu.addEventListener("click", function() {
|
2017-05-06 08:24:01 +02:00
|
|
|
Engine.loadActiveScriptsContent();
|
|
|
|
iTutorialNextStep();
|
|
|
|
return false;
|
|
|
|
});
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.ActiveScriptsPage:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadActiveScriptsContent();
|
2017-05-06 08:24:01 +02:00
|
|
|
iTutorialSetText("This page displays stats/information about all of your scripts that are " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"running across every existing server. You can use this to gauge how well " +
|
2019-05-07 03:01:06 +02:00
|
|
|
"your scripts are doing. Let's go back to the Terminal now using the 'Terminal' " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"link.");
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "none";
|
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Flash 'Terminal' button and set its tutorial click handler
|
2018-08-29 21:06:21 +02:00
|
|
|
terminalMainMenu.setAttribute("class", "flashing-button");
|
|
|
|
terminalMainMenu.addEventListener("click", function() {
|
2017-05-06 08:24:01 +02:00
|
|
|
Engine.loadTerminalContent();
|
|
|
|
iTutorialNextStep();
|
|
|
|
return false;
|
|
|
|
});
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.ActiveScriptsToTerminal:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("One last thing about scripts, each active script contains logs that detail " +
|
|
|
|
"what it's doing. We can check these logs using the 'tail' command. Do that " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"now for the script we just ran by typing 'tail foodnstuff.script'");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // next step triggered by terminal command
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TerminalTailScript:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("The log for this script won't show much right now (it might show nothing at all) because it " +
|
|
|
|
"just started running...but check back again in a few minutes! <br><br>" +
|
|
|
|
"This pretty much covers the basics of hacking. To learn more about writing " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"scripts using the Netscript language, select the 'Tutorial' link in the " +
|
2018-08-29 21:06:21 +02:00
|
|
|
"main navigation menu to look at the documentation. " +
|
|
|
|
"<strong style='background-color:#444;'>If you are an experienced JavaScript " +
|
|
|
|
"developer, I would highly suggest you check out the section on " +
|
|
|
|
"NetscriptJS/Netscript 2.0.</strong><br><br>For now, let's move on to something else!");
|
|
|
|
nextBtn.style.display = "inline-block";
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.GoToHacknetNodesPage:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTerminalContent();
|
2017-05-06 08:24:01 +02:00
|
|
|
iTutorialSetText("Hacking is not the only way to earn money. One other way to passively " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"earn money is by purchasing and upgrading Hacknet Nodes. Let's go to " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"the 'Hacknet Nodes' page through the main navigation menu now.");
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "none";
|
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Flash 'Hacknet' menu and set its tutorial click handler
|
2018-08-29 21:06:21 +02:00
|
|
|
hacknetMainMenu.setAttribute("class", "flashing-button");
|
|
|
|
hacknetMainMenu.addEventListener("click", function() {
|
2017-05-06 08:24:01 +02:00
|
|
|
Engine.loadHacknetNodesContent();
|
|
|
|
iTutorialNextStep();
|
|
|
|
return false;
|
|
|
|
});
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.HacknetNodesIntroduction:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadHacknetNodesContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("From this page you can purchase new Hacknet Nodes and upgrade your " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"existing ones. Let's purchase a new one now.");
|
2019-04-13 03:22:46 +02:00
|
|
|
nextBtn.style.display = "none"; // Next step triggered by purchaseHacknet() (HacknetNode.js)
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.HacknetNodesGoToWorldPage:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadHacknetNodesContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("You just purchased a Hacknet Node! This Hacknet Node will passively " +
|
|
|
|
"earn you money over time, both online and offline. When you get enough " +
|
|
|
|
" money, you can upgrade " +
|
2017-09-05 03:39:00 +02:00
|
|
|
"your newly-purchased Hacknet Node below.<br><br>" +
|
2017-07-22 00:54:55 +02:00
|
|
|
"Let's go to the 'City' page through the main navigation menu.");
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "none";
|
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Flash 'City' menu and set its tutorial click handler
|
2018-08-29 21:06:21 +02:00
|
|
|
cityMainMenu.setAttribute("class", "flashing-button");
|
|
|
|
cityMainMenu.addEventListener("click", function() {
|
2019-04-01 11:23:25 +02:00
|
|
|
Engine.loadLocationContent();
|
2017-05-06 08:24:01 +02:00
|
|
|
iTutorialNextStep();
|
|
|
|
return false;
|
|
|
|
});
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.WorldDescription:
|
2019-04-01 11:23:25 +02:00
|
|
|
Engine.loadLocationContent();
|
2017-05-06 08:24:01 +02:00
|
|
|
iTutorialSetText("This page lists all of the different locations you can currently " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"travel to. Each location has something that you can do. " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"There's a lot of content out in the world, make sure " +
|
2017-08-15 22:22:46 +02:00
|
|
|
"you explore and discover!<br><br>" +
|
2017-05-06 08:24:01 +02:00
|
|
|
"Lastly, click on the 'Tutorial' link in the main navigation menu.");
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "none";
|
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Flash 'Tutorial' menu and set its tutorial click handler
|
2018-08-29 21:06:21 +02:00
|
|
|
tutorialMainMenu.setAttribute("class", "flashing-button");
|
|
|
|
tutorialMainMenu.addEventListener("click", function() {
|
2017-05-06 08:24:01 +02:00
|
|
|
Engine.loadTutorialContent();
|
|
|
|
iTutorialNextStep();
|
|
|
|
return false;
|
|
|
|
});
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.TutorialPageInfo:
|
2018-08-29 21:06:21 +02:00
|
|
|
Engine.loadTutorialContent();
|
2017-08-15 22:22:46 +02:00
|
|
|
iTutorialSetText("This page contains a lot of different documentation about the game's " +
|
|
|
|
"content and mechanics. <strong style='background-color:#444;'> I know it's a lot, but I highly suggest you read " +
|
|
|
|
"(or at least skim) through this before you start playing</strong>. That's the end of the tutorial. " +
|
2017-05-06 08:24:01 +02:00
|
|
|
"Hope you enjoy the game!");
|
2018-08-29 21:06:21 +02:00
|
|
|
nextBtn.style.display = "inline-block";
|
|
|
|
nextBtn.innerHTML = "Finish Tutorial";
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
case iTutorialSteps.End:
|
2017-05-06 08:24:01 +02:00
|
|
|
iTutorialEnd();
|
2017-05-05 23:27:35 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Error("Invalid tutorial step");
|
|
|
|
}
|
2018-08-29 21:06:21 +02:00
|
|
|
|
|
|
|
if (ITutorial.stepIsDone[ITutorial.currStep] === true) {
|
|
|
|
nextBtn.style.display = "inline-block";
|
|
|
|
}
|
2017-05-05 23:27:35 +02:00
|
|
|
}
|
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Go to the next step and evaluate it
|
2017-05-05 23:27:35 +02:00
|
|
|
function iTutorialNextStep() {
|
2019-04-13 03:22:46 +02:00
|
|
|
// Special behavior for certain steps
|
2018-08-29 21:06:21 +02:00
|
|
|
if (ITutorial.currStep === iTutorialSteps.GoToCharacterPage) {
|
2017-07-22 00:54:55 +02:00
|
|
|
document.getElementById("stats-menu-link").removeAttribute("class");
|
2018-08-29 21:06:21 +02:00
|
|
|
}
|
|
|
|
if (ITutorial.currStep === iTutorialSteps.CharacterGoToTerminalPage) {
|
2017-05-15 07:09:14 +02:00
|
|
|
document.getElementById("terminal-menu-link").removeAttribute("class");
|
2018-08-29 21:06:21 +02:00
|
|
|
}
|
|
|
|
if (ITutorial.currStep === iTutorialSteps.TerminalGoToActiveScriptsPage) {
|
2017-05-15 07:09:14 +02:00
|
|
|
document.getElementById("active-scripts-menu-link").removeAttribute("class");
|
2018-08-29 21:06:21 +02:00
|
|
|
}
|
|
|
|
if (ITutorial.currStep === iTutorialSteps.ActiveScriptsPage) {
|
2017-05-15 07:09:14 +02:00
|
|
|
document.getElementById("terminal-menu-link").removeAttribute("class");
|
2018-08-29 21:06:21 +02:00
|
|
|
}
|
|
|
|
if (ITutorial.currStep === iTutorialSteps.GoToHacknetNodesPage) {
|
2017-05-15 07:09:14 +02:00
|
|
|
document.getElementById("hacknet-nodes-menu-link").removeAttribute("class");
|
2018-08-29 21:06:21 +02:00
|
|
|
}
|
|
|
|
if (ITutorial.currStep === iTutorialSteps.HacknetNodesGoToWorldPage) {
|
2017-07-22 00:54:55 +02:00
|
|
|
document.getElementById("city-menu-link").removeAttribute("class");
|
2018-08-29 21:06:21 +02:00
|
|
|
}
|
|
|
|
if (ITutorial.currStep === iTutorialSteps.WorldDescription) {
|
2017-05-15 07:09:14 +02:00
|
|
|
document.getElementById("tutorial-menu-link").removeAttribute("class");
|
2017-05-05 23:27:35 +02:00
|
|
|
}
|
2018-08-29 21:06:21 +02:00
|
|
|
|
|
|
|
ITutorial.stepIsDone[ITutorial.currStep] = true;
|
|
|
|
if (ITutorial.currStep < iTutorialSteps.End) {
|
|
|
|
ITutorial.currStep += 1;
|
|
|
|
}
|
|
|
|
iTutorialEvaluateStep();
|
2017-05-05 23:27:35 +02:00
|
|
|
}
|
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Go to previous step and evaluate
|
2017-05-15 07:09:14 +02:00
|
|
|
function iTutorialPrevStep() {
|
2018-08-29 21:06:21 +02:00
|
|
|
if (ITutorial.currStep > iTutorialSteps.Start) {
|
|
|
|
ITutorial.currStep -= 1;
|
2017-05-15 07:09:14 +02:00
|
|
|
}
|
2018-08-29 21:06:21 +02:00
|
|
|
iTutorialEvaluateStep();
|
2017-05-15 07:09:14 +02:00
|
|
|
}
|
|
|
|
|
2017-05-05 23:27:35 +02:00
|
|
|
function iTutorialEnd() {
|
2019-04-13 03:22:46 +02:00
|
|
|
// Re-enable auto save
|
2018-08-29 21:06:21 +02:00
|
|
|
if (Settings.AutosaveInterval === 0) {
|
|
|
|
Engine.Counters.autoSaveCounter = Infinity;
|
|
|
|
} else {
|
|
|
|
Engine.Counters.autoSaveCounter = Settings.AutosaveInterval * 5;
|
|
|
|
}
|
|
|
|
|
2019-01-30 07:02:27 +01:00
|
|
|
// Initialize references to main menu links
|
|
|
|
// We have to call initializeMainMenuLinks() again because the Interactive Tutorial
|
|
|
|
// re-creates Main menu links with clearEventListeners()
|
|
|
|
if (!initializeMainMenuLinks()) {
|
|
|
|
const errorMsg = "Failed to initialize Main Menu Links. Please try refreshing the page. " +
|
|
|
|
"If that doesn't work, report the issue to the developer";
|
|
|
|
exceptionAlert(new Error(errorMsg));
|
|
|
|
console.error(errorMsg);
|
|
|
|
return;
|
|
|
|
}
|
2017-05-06 08:24:01 +02:00
|
|
|
Engine.init();
|
2019-01-30 07:02:27 +01:00
|
|
|
|
2018-08-29 21:06:21 +02:00
|
|
|
ITutorial.currStep = iTutorialSteps.End;
|
|
|
|
ITutorial.isRunning = false;
|
2017-05-06 08:24:01 +02:00
|
|
|
document.getElementById("interactive-tutorial-container").style.display = "none";
|
2018-08-29 21:06:21 +02:00
|
|
|
|
2019-04-13 03:22:46 +02:00
|
|
|
// Create a popup with final introductory stuff
|
2018-08-29 21:06:21 +02:00
|
|
|
var popupId = "interactive-tutorial-ending-popup";
|
|
|
|
var txt = createElement("p", {
|
|
|
|
innerHTML:
|
|
|
|
"If you are new to the game, the following links may be useful for you!<br><br>" +
|
2019-03-03 04:08:54 +01:00
|
|
|
"<a class='a-link-button' href='https://bitburner.readthedocs.io/en/latest/guidesandtips/gettingstartedguideforbeginnerprogrammers.html' target='_blank'>Getting Started Guide</a>" +
|
2018-08-29 21:06:21 +02:00
|
|
|
"<a class='a-link-button' href='https://bitburner.readthedocs.io/en/latest/' target='_blank'>Documentation</a><br><br>" +
|
|
|
|
"The Beginner's Guide to Hacking was added to your home computer! It contains some tips/pointers for starting out with the game. " +
|
2021-04-30 05:52:56 +02:00
|
|
|
"To read it, go to Terminal and enter<br><br>cat " + LiteratureNames.HackersStartingHandbook,
|
2018-08-29 21:06:21 +02:00
|
|
|
});
|
|
|
|
var gotitBtn = createElement("a", {
|
|
|
|
class:"a-link-button", float:"right", padding:"6px", innerText:"Got it!",
|
|
|
|
clickListener:()=>{
|
|
|
|
removeElementById(popupId);
|
2021-04-30 05:52:56 +02:00
|
|
|
},
|
2018-08-29 21:06:21 +02:00
|
|
|
});
|
|
|
|
createPopup(popupId, [txt, gotitBtn]);
|
|
|
|
|
2021-03-14 07:08:24 +01:00
|
|
|
Player.getHomeComputer().messages.push(LiteratureNames.HackersStartingHandbook);
|
2017-05-05 23:27:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function iTutorialSetText(txt) {
|
|
|
|
var textBox = document.getElementById("interactive-tutorial-text");
|
|
|
|
if (textBox == null) {throw new Error("Could not find text box"); return;}
|
|
|
|
textBox.innerHTML = txt;
|
2017-07-03 21:42:11 +02:00
|
|
|
textBox.parentElement.scrollTop = 0; // this resets scroll position
|
2017-08-15 22:22:46 +02:00
|
|
|
}
|
2017-08-30 19:44:29 +02:00
|
|
|
|
2018-08-29 21:06:21 +02:00
|
|
|
export {iTutorialSteps, iTutorialEnd, iTutorialStart, iTutorialNextStep, ITutorial};
|