Denizen API Command Template: Difference between revisions

From Citizens Wiki

No edit summary
 
(13 intermediate revisions by one other user not shown)
Line 1: Line 1:
=== YourCommand extends AbstractCommand ===


Below is a command template, ready to copy/paste into your project. You can also view a copy on [http://pastie.org/4368657 Pastie], which is color coded.
For more up-to-date information and full details on specific features (individual commands or tags, for example), check the [https://meta.denizenscript.com/ Meta Documentation].
 
If you want a full tutorial to help get you set up, check out the [https://guide.denizenscript.com/ Beginner's Guide] text website.
 
If you need quick help, visit our [https://discord.gg/Q6pZGSR Discord group].
 
<br><br><br>
 
<span style="font-family:natalya-alternate-one; font-size:300%; margin-right:-7px; margin-left:-10px;">This wiki is outdated, please view the tutorial videos/guide, meta documentation, or Discord group (all linked above) for up-to-date information!</span>
 
<br><br><br>
 
 
 
 
 
 
 
 
 
== Custom Commands ==
<div style="margin-right:2.0em; padding:10px">
Denizen is extensible! In addition to Triggers and Requirements(soon!), an API is being provided to help you make custom Script Commands in your project! Have a neat idea for a command? Have a better way to STRIKE a player? (MEGASTRIKE command, anybody?)
 
Keep reading!
</div>
 
== YourCommand extends AbstractCommand ==
<div style="margin-right:2.0em; padding:10px">
The first thing you need is a custom command class that extends the Denizen AbstractTrigger. This contains the method called when it's time for your command to be executed. This is called every-single-time a script contains your command.
</div>
 
==== Example Command ====
<div style="margin-right:2.0em; padding:10px">
Sometimes the easiest way to work is to see an actual example. View a copy of the STRIKE command on Pastie.org for a very simple command. For examples, you can always check [https://github.com/aufdemrand/Denizen/tree/master/src/main/java/net/aufdemrand/denizen/commands/core Core Denizen Commands].
</div>


==== Command Template ====
==== Command Template ====
<div style="margin-right:2.0em; padding:10px">
[http://pastie.org/4368657 View a command template copy on Pastie.org] for a copy/paste of the skeleton of a custom denizen commmand.
</div>
=== The ScriptEntry object ===
<div style="margin-right:2.0em; padding:10px">
When Denizen reads scripts, each line is turned into a SciptScript object that contains the command, the arguments, and various other data and objects, as described below. Your command is an extension of this AbstractCommand class, and each time your command is called in for execution, it's this class that is executed and handled a ScriptEntry. How your Command uses the ScriptEntry information is of course, up to you.
The Denizen ScriptEngine and Executer construct and fill a ScriptEntry with information that can be used in your command. Below is a list of methods you can use against the scriptEntry sent to your commands execute method.


<pre>
{{ScriptEntry
package net.aufdemrand.denizen.commands.core;
|1= getCommand()
|2= String value of the name of the command
|3= Used by default in alerting the user when denizen is in debug mode, if the template is used. Also could be useful if your command module contains instructions for more than one command. You can register multiple commands to your module, as discussed later in this document.
}}
 
{{ScriptEntry
|1= arguments()
|2= String[] value of the arguments used in the script
|3= This is the output of Denizen's argBuilder method. It splits the command arguments up with spaces and quotes. For instance: <code>arg1 arg2 'argument 3' "argument 4" would return a string array with a size of 4.
}}


import java.util.logging.Level;
{{ScriptEntry
|1= arguments()
|2= String[] value of the arguments used in the script
|3= This is the output of Denizen's argBuilder method. It splits the command arguments up with spaces and quotes. For instance: <code>arg1 arg2 'this is argument 3' "and that's argument 4" would return a string array with a size of 4. Note argument 4, which handles an argument using a single quote. Hint: Double quotes around the argument, and vice-versa.
}}


import org.bukkit.Location;
{{ScriptEntry
|1= sendingQueue()
|2= QueueType of either QueueType.TRIGGER, QueueType.TASK, or QueueType.ACTIVITY
|3= Denizen's Script Engine contains a method called runQeues which sends the ScriptEntries off to the executer. During this process, it stores which queue the command was called from. Can be used in advanced situations. Typically, if called by an Interact Script, this contains QueueType.TRIGGER.
}}


import net.aufdemrand.denizen.bookmarks.BookmarkHelper.BookmarkType;
{{ScriptEntry
import net.aufdemrand.denizen.commands.AbstractCommand;
|1= getScript()
import net.aufdemrand.denizen.scripts.ScriptEntry;
|2= String value of the script name
import net.citizensnpcs.command.exception.CommandException;
|3= Could be useful if storing information about a Script. For example, Denizen's FINISH command takes this information for granted if no script is otherwise indicated in the arguments.
}}


/**
{{ScriptEntry
* Your command!
|1= getStep()
* This class is a template for a Command in Denizen.
|2= Integer value of the script's step
*
|3= Could be used similarly to getScript(). The ZAP command uses this information to know which step to proceed to.
* @author You!
}}
*/


public class SampleCommand extends AbstractCommand {
{{ScriptEntry
|1= getPlayer()
|2= Player object of the script triggerer
|3= Pretty straight forward. The bukkit Player object can get all kinds of information about the player.
}}


/* COMMAND_NAME [TYPICAL] (ARGUMENTS) */
{{ScriptEntry
|1= getDenizen()
|2= DenizenNPC object of the script triggeree
|3= DenizenNPC makes it easy to work with the Denizen that has been interacted with. Good example: LOOK command.
}}


/*
{{ScriptEntry
* Arguments: [] - Required, () - Optional
|1= getInitiatedTime()
* [TYPICAL] argument with a description if necessary.
|2= Long value of System.currentTimeMillis()
* (ARGUMENTS) should be clear and concise.
|3= Contains a <code>Long System.currentTimeMillis()</code> of the system time in which it was sent to your Command for processing which you can use to check against.
}}
* Modifiers:
* (MODIFIER:VALUE) These are typically advanced usage arguments.
* (DURATION:#) They should always be optional. Use standard modifiers
*  already established if at all possible.
* Example Usage:
* COMMAND_NAME VALUE
* COMMAND_NAME DIFFERENTVALUE OPTIONALVALUE
* COMMAND_NAME ANOTHERVALUE 'MODIFIER:Show one-line examples.'
*
*/


@SuppressWarnings("unused") // This should be removed in your code.
{{ScriptEntry
@Override
|1= setDelayedTime()  
|2= should be a Long value in the format of System.currentTimeMillis()
// This is the method that is called when your command is ready to be executed.
|3= Advanced usage. See the DELAY command for an example.
public boolean execute(ScriptEntry theEntry) throws CommandException {
}}


/* Initialize variables */
{{ScriptEntry
|1= getDelayedTime()
|2= Long value that was set with setDelayedTime()
|3= Advanced usage. Again, see the DELAY command for an example. If not set with setDelayedTime(), this is null by default.
}}


    // Typically initialized as null and filled as needed. Remember: theEntry
{{ScriptEntry
    // contains some information passed through the execution process.
|1= getTexts()
Boolean requiredVariable = null;
|2= String[2] of text sent by a Trigger
Location sampleBookmark = null;
|3= Vanilla Denizen will use this to store information sent to/from a Chat Trigger. Element [0] contains the text typed by the Player when the script was triggered, and element [1] contains the text as set in the Trigger: node of the script.  
}}
/* Match arguments to expected variables */
</div>
if (theEntry.arguments() != null) {
for (String thisArgument : theEntry.arguments()) {
// Do this routine for each argument supplied.
if (plugin.debugMode) plugin.getLogger().info("Processing command " + theEntry.getCommand() + " argument: " + thisArgument);


// Includes are some typical arguments. Modify/add code to handle your command needs.
// If argument is a number.
if (thisArgument.matches("((-|\\+)?[0-9]+(\\.[0-9]+)?)+")) {
// Insert code here.


}
== Registering your command ==
<div style="margin-right:2.0em; padding:10px">
// If argument is a valid bookmark, set location.
In order for Denizen to use your command, it needs to be registered. This could probably be done in your plugin's onEnable() method.
else if (plugin.bookmarks.exists(theEntry.getDenizen(), thisArgument)) {
</div>
if (plugin.debugMode) plugin.getLogger().log(Level.INFO, "...matched bookmark '" + thisArgument + "'.");
sampleBookmark = plugin.bookmarks.get(theEntry.getDenizen(), thisArgument, BookmarkType.LOCATION);
} else if (thisArgument.split(":").length == 2) {
if (plugin.bookmarks.exists(thisArgument.split(":")[0], thisArgument.split(":")[1]))
if (plugin.debugMode) plugin.getLogger().log(Level.INFO, "...matched bookmark '" + thisArgument.split(":")[0] + "'.");
sampleBookmark = plugin.bookmarks.get(thisArgument.split(":")[0], thisArgument.split(":")[1], BookmarkType.LOCATION);
}
// If argument is a modifier.
else if (thisArgument.toUpperCase().contains("MODIFIER:")) {
if (plugin.debugMode) plugin.getLogger().log(Level.INFO, "...matched modifier '" + thisArgument.split(":")[0].toUpperCase() + "'.");


// Insert code here.
==== Example code ====
<div style="margin-right:2.0em; padding:10px">
}
You can also see a [http://pastie.org/4369286 formatted version on Pastie.org].
// If can't match to anything...
// This isn't always possible, depending on the arguments your command uses, but nice if you can.
else {
if (plugin.debugMode) plugin.getLogger().log(Level.INFO, "...unable to match argument!");
}
}
}


/* Execute the command, if all required variables are filled. */
<pre>
if (requiredVariable != null) {
/*
* Sets up my plugin on start of the craftbukkit server.
*/
// Execution process.
// Do whatever you want the command to do, here.
/* Command has sucessfully finished */
return true;
}
// else...
/* Error processing */
// Processing has gotten to here, there's probably not been enough arguments.
// Let's alert the console.
if (plugin.debugMode) if (theEntry.arguments() == null)
throw new CommandException("...not enough arguments! Usage: SAMPLECOMMAND [TYPICAL] (ARGUMENTS)");
return false;
}


  // You can include more methods in this class if necessary. Or not. :)
public MyCommand myCommand = new MyCommand();


@Override
public void onEnable() {
  getLogger().info("Enabling my plugin!");
 
  try {
      myCommand.registerAs("CUSTOM_COMMAND");
  } catch (ActivationException e) {
      plugin.getLogger().log(Level.SEVERE, "Oh no! Activation Exception!");
      e.printStackTrace();
  }
}
}
</pre>
</pre>

Latest revision as of 18:56, 3 September 2021

For more up-to-date information and full details on specific features (individual commands or tags, for example), check the Meta Documentation.

If you want a full tutorial to help get you set up, check out the Beginner's Guide text website.

If you need quick help, visit our Discord group.




This wiki is outdated, please view the tutorial videos/guide, meta documentation, or Discord group (all linked above) for up-to-date information!








Custom Commands

Denizen is extensible! In addition to Triggers and Requirements(soon!), an API is being provided to help you make custom Script Commands in your project! Have a neat idea for a command? Have a better way to STRIKE a player? (MEGASTRIKE command, anybody?)

Keep reading!

YourCommand extends AbstractCommand

The first thing you need is a custom command class that extends the Denizen AbstractTrigger. This contains the method called when it's time for your command to be executed. This is called every-single-time a script contains your command.

Example Command

Sometimes the easiest way to work is to see an actual example. View a copy of the STRIKE command on Pastie.org for a very simple command. For examples, you can always check Core Denizen Commands.

Command Template

View a command template copy on Pastie.org for a copy/paste of the skeleton of a custom denizen commmand.

The ScriptEntry object

When Denizen reads scripts, each line is turned into a SciptScript object that contains the command, the arguments, and various other data and objects, as described below. Your command is an extension of this AbstractCommand class, and each time your command is called in for execution, it's this class that is executed and handled a ScriptEntry. How your Command uses the ScriptEntry information is of course, up to you.

The Denizen ScriptEngine and Executer construct and fill a ScriptEntry with information that can be used in your command. Below is a list of methods you can use against the scriptEntry sent to your commands execute method.

getCommand()
String value of the name of the command
Used by default in alerting the user when denizen is in debug mode, if the template is used. Also could be useful if your command module contains instructions for more than one command. You can register multiple commands to your module, as discussed later in this document.
arguments()
String[] value of the arguments used in the script
This is the output of Denizen's argBuilder method. It splits the command arguments up with spaces and quotes. For instance: arg1 arg2 'argument 3' "argument 4" would return a string array with a size of 4.
arguments()
String[] value of the arguments used in the script
This is the output of Denizen's argBuilder method. It splits the command arguments up with spaces and quotes. For instance: arg1 arg2 'this is argument 3' "and that's argument 4" would return a string array with a size of 4. Note argument 4, which handles an argument using a single quote. Hint: Double quotes around the argument, and vice-versa.
sendingQueue()
QueueType of either QueueType.TRIGGER, QueueType.TASK, or QueueType.ACTIVITY
Denizen's Script Engine contains a method called runQeues which sends the ScriptEntries off to the executer. During this process, it stores which queue the command was called from. Can be used in advanced situations. Typically, if called by an Interact Script, this contains QueueType.TRIGGER.
getScript()
String value of the script name
Could be useful if storing information about a Script. For example, Denizen's FINISH command takes this information for granted if no script is otherwise indicated in the arguments.
getStep()
Integer value of the script's step
Could be used similarly to getScript(). The ZAP command uses this information to know which step to proceed to.
getPlayer()
Player object of the script triggerer
Pretty straight forward. The bukkit Player object can get all kinds of information about the player.
getDenizen()
DenizenNPC object of the script triggeree
DenizenNPC makes it easy to work with the Denizen that has been interacted with. Good example: LOOK command.
getInitiatedTime()
Long value of System.currentTimeMillis()
Contains a Long System.currentTimeMillis() of the system time in which it was sent to your Command for processing which you can use to check against.
setDelayedTime()
should be a Long value in the format of System.currentTimeMillis()
Advanced usage. See the DELAY command for an example.
getDelayedTime()
Long value that was set with setDelayedTime()
Advanced usage. Again, see the DELAY command for an example. If not set with setDelayedTime(), this is null by default.
getTexts()
String[2] of text sent by a Trigger
Vanilla Denizen will use this to store information sent to/from a Chat Trigger. Element [0] contains the text typed by the Player when the script was triggered, and element [1] contains the text as set in the Trigger: node of the script.


Registering your command

In order for Denizen to use your command, it needs to be registered. This could probably be done in your plugin's onEnable() method.

Example code

You can also see a formatted version on Pastie.org.

/*
 * Sets up my plugin on start of the craftbukkit server.	
 */

public MyCommand myCommand = new MyCommand();

@Override
public void onEnable() {
   getLogger().info("Enabling my plugin!");
   
   try {
       myCommand.registerAs("CUSTOM_COMMAND");
   } catch (ActivationException e) {
       plugin.getLogger().log(Level.SEVERE, "Oh no! Activation Exception!");
       e.printStackTrace();
   }
}