Skip to content

Commit a2ba252

Browse files
committed
Initial commit
0 parents  commit a2ba252

File tree

8 files changed

+306
-0
lines changed

8 files changed

+306
-0
lines changed

.gitignore

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
# Created by .ignore support plugin (hsz.mobi)
2+
### Maven template
3+
target/
4+
pom.xml.tag
5+
pom.xml.releaseBackup
6+
pom.xml.versionsBackup
7+
pom.xml.next
8+
release.properties
9+
dependency-reduced-pom.xml
10+
buildNumber.properties
11+
.mvn/timing.properties
12+
# https://github.com/takari/maven-wrapper#usage-without-binary-jar
13+
.mvn/wrapper/maven-wrapper.jar
14+
15+
.idea
16+
*.iml
17+
.project
18+
/run

LICENSE

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
This is free and unencumbered software released into the public domain.
2+
3+
Anyone is free to copy, modify, publish, use, compile, sell, or
4+
distribute this software, either in source code form or as a compiled
5+
binary, for any purpose, commercial or non-commercial, and by any
6+
means.
7+
8+
In jurisdictions that recognize copyright laws, the author or authors
9+
of this software dedicate any and all copyright interest in the
10+
software to the public domain. We make this dedication for the benefit
11+
of the public at large and to the detriment of our heirs and
12+
successors. We intend this dedication to be an overt act of
13+
relinquishment in perpetuity of all present and future rights to this
14+
software under copyright law.
15+
16+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
19+
IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
20+
OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
21+
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
22+
OTHER DEALINGS IN THE SOFTWARE.
23+
24+
For more information, please refer to <https://unlicense.org>

README.md

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
# Example Plugin for PowerNukkit
2+
This is an example plugin which can also be used as template to start your own plugin.
3+
4+
As an example I created a plugin named clone-me, it creates a clone of yourself when you run `/clone`
5+
and gives you a flower if you hit the clone and then despawn the clone. It also send some fancy messages.
6+
7+
These is enough to serve as an example on how to:
8+
- Begin a new plugin
9+
- Create event listeners and handlers
10+
- Create custom commands
11+
- Format text
12+
- Spawn NPCs
13+
- Despawn NPCs
14+
- Detect attacks
15+
- Make entities invulnerable
16+
- Create and fill a `plugin.yml` file
17+
- Debug your plugin properly
18+
19+
## Cloning and importing
20+
1. Just do a normal `git clone https://github.com/PowerNukkit/ExamplePlugin.git` (or the URL of your own git repository)
21+
2. Import the `pom.xml` file with your IDE, it should do the rest by itself
22+
23+
## Debugging
24+
1. Create a zip file containing only your `plugin.yml` file
25+
2. Rename the zip file to change the extension to jar
26+
3. Create an empty folder anywhere, that will be your server folder.
27+
<small>_Note: You don't need to place the PowerNukkit jar in the folder, your IDE will load it from the maven classpath._</small>
28+
4. Create a folder named `plugins` inside your server folder
29+
<small>_Note: It is needed to bootstrap your plugin, your IDE will load your plugin classes from the classpath automatically,
30+
so it needs to have only the `plugin.yml` file._</small>
31+
5. Move the jar file that contains only the `plugin.yml` to the `plugins` folder
32+
6. Create a new Application run configuration setting the working directory to the server folder and the main class to: `cn.nukkit.Nukkit`
33+
![](https://i.imgur.com/NUrrZab.png)
34+
7. Now you can run in debug mode. If you change the `plugin.yml` you will need to update the jar file that you've made.

pom.xml

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
2+
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
3+
<modelVersion>4.0.0</modelVersion>
4+
5+
<name>ExamplePlugin</name>
6+
<artifactId>example-plugin</artifactId>
7+
<groupId>org.powernukkit.plugins</groupId>
8+
<version>0.1.0-SNAPSHOT</version>
9+
<packaging>jar</packaging>
10+
11+
<dependencies>
12+
<dependency>
13+
<groupId>org.powernukkit</groupId>
14+
<artifactId>powernukkit</artifactId>
15+
<version>1.3.1.4-PN</version>
16+
</dependency>
17+
</dependencies>
18+
19+
<properties>
20+
<maven.compiler.target>1.8</maven.compiler.target>
21+
<maven.compiler.source>1.8</maven.compiler.source>
22+
</properties>
23+
</project>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
package org.powernukkit.plugins.example;
2+
3+
import cn.nukkit.Player;
4+
import cn.nukkit.command.Command;
5+
import cn.nukkit.command.CommandExecutor;
6+
import cn.nukkit.command.CommandSender;
7+
import cn.nukkit.entity.EntityHuman;
8+
import cn.nukkit.nbt.tag.CompoundTag;
9+
import cn.nukkit.utils.TextFormat;
10+
11+
public class CloneCommand implements CommandExecutor {
12+
@Override
13+
public boolean onCommand(CommandSender commandSender, Command command, String s, String[] strings) {
14+
if (!(commandSender instanceof Player)) {
15+
commandSender.sendMessage(TextFormat.DARK_RED+""+TextFormat.BOLD + "Error!"+TextFormat.RESET+TextFormat.RED+" Only players can execute this command!");
16+
return false;
17+
}
18+
Player player = (Player) commandSender;
19+
player.saveNBT();
20+
EntityHuman human = new EntityHuman(player.getChunk(), new CompoundTag()
21+
.put("Pos", player.namedTag.get("Pos").copy())
22+
.put("Rotation", player.namedTag.get("Rotation").copy())
23+
.put("Motion", player.namedTag.get("Motion").copy())
24+
.put("Skin", player.namedTag.get("Skin").copy())
25+
.putBoolean("IsCloned", true)
26+
);
27+
human.setSkin(player.getSkin());
28+
human.spawnToAll();
29+
player.sendMessage(TextFormat.DARK_GREEN+""+TextFormat.BOLD+"Success!"+TextFormat.RESET+TextFormat.GREEN+" You have been cloned!");
30+
return true;
31+
}
32+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
package org.powernukkit.plugins.example;
2+
3+
import cn.nukkit.Player;
4+
import cn.nukkit.block.BlockID;
5+
import cn.nukkit.entity.Entity;
6+
import cn.nukkit.entity.EntityHuman;
7+
import cn.nukkit.event.EventHandler;
8+
import cn.nukkit.event.Listener;
9+
import cn.nukkit.event.entity.EntityDamageByEntityEvent;
10+
import cn.nukkit.item.Item;
11+
import cn.nukkit.utils.TextFormat;
12+
13+
public class CloneListener implements Listener {
14+
@EventHandler
15+
public void onCloneDamage(EntityDamageByEntityEvent event) {
16+
Entity entity = event.getEntity();
17+
18+
// Affect only the clones
19+
if (!(entity instanceof EntityHuman)
20+
|| !entity.namedTag.getBoolean("IsCloned")) {
21+
return;
22+
}
23+
24+
// Makes the clones invulnerable to non-player damage
25+
if (!(event.getDamager() instanceof Player)) {
26+
event.setCancelled();
27+
return;
28+
}
29+
30+
// If the clone is hit by a player, despawn it
31+
Player player = (Player) event.getDamager();
32+
player.sendMessage(TextFormat.GOLD+""+TextFormat.BOLD+"WOW!"+TextFormat.RESET+TextFormat.YELLOW+" You found a clone!");
33+
entity.close();
34+
35+
// And give the player a present
36+
Item flowerItem = Item.getBlock(BlockID.RED_FLOWER);
37+
flowerItem.setCustomName(TextFormat.RESET+""+TextFormat.RED+"Congratulations!");
38+
flowerItem.setLore(TextFormat.RESET+""+TextFormat.LIGHT_PURPLE+"This is a present for your finding!");
39+
40+
// The will guarantee that the player receive the present by dropping it on the floor if the inventory is full
41+
for (Item drop: player.getInventory().addItem(flowerItem)) {
42+
player.getLevel().dropItem(player, drop);
43+
}
44+
}
45+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
package org.powernukkit.plugins.example;
2+
3+
import cn.nukkit.command.Command;
4+
import cn.nukkit.command.CommandSender;
5+
import cn.nukkit.command.PluginCommand;
6+
import cn.nukkit.plugin.PluginBase;
7+
8+
public class CloneMePlugin extends PluginBase {
9+
@Override
10+
public void onEnable() {
11+
getLogger().info("Hello world! :D");
12+
if (System.getProperty("os.name").startsWith("Windows")) {
13+
getLogger().warning("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!");
14+
getLogger().warning("!!! ATTENTION WINDOWS USER !!!");
15+
getLogger().warning("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!");
16+
getLogger().warning("To connect to the server in localhost you must allow the game to access localhost:");
17+
getLogger().warning("1. Open PowerShell as admin");
18+
getLogger().warning("2. Run this command: CheckNetIsolation LoopbackExempt -a -n=\"Microsoft.MinecraftUWP_8wekyb3d8bbwe\"");
19+
getLogger().warning("3. Restart your computer (if needed, try to restart your game first)");
20+
getLogger().warning("This issue occurs due to loopback restrictions on Windows 10 UWP apps");
21+
}
22+
23+
getLogger().info("TIP: Make sure your break points are set to pause ONLY THE THREAD and NOT ALL THREADS!");
24+
getLogger().info("https://imgur.com/ygwen76");
25+
getLogger().info("If you do this, you won't get disconnected when you hit a break point");
26+
27+
getLogger().info("TIP: If you are using IntelliJ, use Ctrl+F9 (Build Project) to apply non-structural java changes without restart");
28+
29+
// TODO: Make it easier
30+
// This make the command be executed in a separated class, you need to choose if you want
31+
// it being executed there or in the onCommand bellow, you can't use both
32+
// Simple commands are fine in onCommand but complex command might be more organized
33+
// in their own classes. Also make sure you register this command in plugin.yml
34+
((PluginCommand<?>) getCommand("cloneme")).setExecutor(new CloneCommand());
35+
36+
// You must register your listeners to capture events
37+
// You can make this class implement the Listener itself and invoke registerEvents(this, this)
38+
// But again, if the listener gets too complicated it might be better to group them in different classes
39+
getServer().getPluginManager().registerEvents(new CloneListener(), this);
40+
}
41+
42+
@Override
43+
public void onDisable() {
44+
getLogger().info("Goodbye world :(");
45+
}
46+
47+
@Override
48+
public boolean onCommand(CommandSender sender, Command command, String label, String[] args) {
49+
// You can also override this command instead of setting an executor in onEnable if you prefer
50+
return false;
51+
}
52+
}

src/main/resources/plugin.yml

+78
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,78 @@
1+
# Required
2+
# The final name of your plugin, other plugins may use this name to declare dependency on your plugin
3+
name: CloneMe
4+
5+
# Required
6+
# The name of your class file that overrides PluginBase
7+
main: org.powernukkit.plugins.example.CloneMePlugin
8+
9+
# Required
10+
# The version of your plugin, it's recommended to follow the https://semver.org/ standard
11+
version: "1.0.0"
12+
13+
# Required
14+
# The minimum Cloudburst Nukkit API version, it's not the PowerNukkit version!
15+
# Setting an outdated version usually don't cause issues so it's not a thing to worry too much
16+
api: ["1.0.0"]
17+
18+
# Optional
19+
# At which time your plugin will load.
20+
# Valid options are: POSTWORLD, STARTUP
21+
# Default value is POSTWORLD
22+
load: POSTWORLD
23+
24+
# Optional
25+
# Your name or your organization name or how people identify you
26+
author: Nukkit Project
27+
28+
# Optional
29+
# Explain in few words what this plugin does
30+
description: Example plugin showing the API
31+
32+
# Optional
33+
# A link where the admins can find more info about your plugin
34+
website: https://github.com/Nukkit/ExamplePlugin
35+
36+
# Optional
37+
# Every sub-item must be a block, the key is the command name, you can create as many commands as you want
38+
# Make sure to keep the exact same alignment, it is important
39+
commands:
40+
# The key is how the command will be used, like /cloneme . Avoid uppercase, the client game don't like it so much and may crash.
41+
cloneme:
42+
# Optional
43+
# This information will be displayed in /help
44+
description: Example command
45+
46+
# Optional
47+
# This information will be displayed in /help and when your command executor returns false
48+
usage: "/example"
49+
50+
# Optional
51+
# The permission required to use your command. It's a good practice to always define one even if the command can be used by everybody
52+
# More information below
53+
permission: cloneme.cmd.use
54+
55+
# Optional
56+
# Although you don't need to register your permissions here for them to work, it's good to allow the server owners to see all them quicker
57+
# This section also allows to customize their behavior and provide more information about them
58+
# You can also create group of permission
59+
# Also, every sub-item must be a block just like the commands block above
60+
permissions:
61+
# The key must be the same value you use to define the permission
62+
# It's a good practice to prefix it with your plugin name and create segmentation and then group them to make the permission management easier
63+
cloneme.cmd.use:
64+
# Optional
65+
# This can be used by permission management plugins to show details about the permission key
66+
description: "Allows the user to run the example command"
67+
68+
# Optional
69+
# If a player don't have an allow-deny definition for this key, this default value will be used
70+
# Valid options:
71+
# - true: the permission is granted by default to everybody
72+
# - false: the permission is revoked by default to everybody, including admins and OP
73+
# - op: the permission is revoked by default to everybody, but is granted to OP players
74+
# - notop: the permission is granted by default to everybody, but is revoked to OP players
75+
# Default value is false
76+
default: true
77+
78+
# TODO Add all possible configuration here

0 commit comments

Comments
 (0)