First off, thank you for your interest in contributing to foundry-vtt-dnd5e-types—we need all the help we can get!
The following is a set of guidelines for contributing to foundry-vtt-dnd5e-types. These are mostly guidelines, not rules. Use your best judgment, and feel free to propose changes to this document in a pull request.
- Documentation: README.md, Wiki
- Issue tracker: Issues
- Milestones
- Communication: League Discord (ask for the TypeScript related channels if you're not already in them)
This section guides you through submitting a bug report for foundry-vtt-dnd5e-types. Following these guidelines helps maintainers and the community understand your report and reproduce the behavior.
- Make sure you are using the latest version of foundry-vtt-dnd5e-types for the corresponding Foundry VTT Version.
- Check if the problem has already been reported in our issue tracker.
Bugs are tracked in our issue tracker. When creating a bug report, please provide the following information by using the template.
- Provide the exact version of Foundry VTT and foundry-vtt-dnd5e-types that exhibits the problem.
- Provide minimal steps to reproduce the behavior.
- Describe the problem in a clear and concise way. This includes the actual behavior.
- Describe the expected behavior.
- Optionally, add any additional context if it makes sense.
You can also suggest ideas for the project beyond plain Foundry VTT types. The guidelines are similar to those for bug reporting. The most important one being: Use the template to give all the relevant information about your request.
If you are unsure where to start, you can look through issues tagged with help wanted. These are issues that we would like to have help with.
We also recommend to take a look at the corresponding project board to make sure that nobody is working on your issue yet. Additionally, please let us know if you start working on a specific issue (by writing a comment in that issue) so that we can avoid doing duplicate work.
We use a very simple branching model. We have a branch for every supported dnd5e system version called dnd5e-<version>
(e.g. dnd5e-1.2.4
). All changes for the type definitions for a given system version need to be made through Pull Requests towards the corresponding branch.
When creating a pull request, please provide the following information:
- What issue does this pull request correspond to (if any)?
- What does this pull request implement?
- What changes are made in this pull request?
We use prettier to automatically format the code and have ESlint check for the correct formatting. This even runs automatically as a pre-commit hook so usually you don't have to care for much here. However, there is one additional style guideline that prettier doesn't enforce for us:
In TSDoc comments, always make sure to align the -
for all @param
s.
In very rare occasions, it is acceptable to disable prettier for a specific part of the code to improve the formatting manually, e.g. for deeply nested conditional types, which are simply unreadable if we let prettier format them. See Prettier – Ignoring Code to find out how to do this.
- Try to match the source code of the system as closely as possible in your type definitions. In particular, the order of declarations should be exactly the same. This allows for easy side by side viewing of system files and the type definitions, making the life of code reviewers much easier :)
- Try not to pollute the global namespace with custom types that are not declared by the system itself (typedefs from
foundry.js
should be declared). Instead use a namespace named like the related class and put your custom type in there. - Every class has its own file. The files are structured by class hierarchy. Declarations for code that is not a class belongs directly in the
dnd5e
folder. - Utility types not defined in foundry belong in
types/utils.d.ts
If the type of the options is different from that of the parent class, provide a new Options
interface
in the namespace
of the class you are adding types for. It makes sense to add the default values in that type via @defaultValue
TSDoc. Use this interface as return type for for getDefaultOptions
.
If the type does not differ, just use the Options
interface from the parent class.
In both cases, provide the default value for getDefaultOptions
in TSDoc.
Example:
declare class ActorSheet<
D extends object = ActorSheet.Data<Actor>,
O extends Actor = D extends ActorSheet.Data<infer T> ? T : Actor
> extends BaseEntitySheet<D, O> {
/* ... */
/**
* @override
* @defaultValue
* ```typescript
* mergeObject(super.defaultOptions, {
* height: 720,
* width: 800,
* template: "templates/sheets/actor-sheet.html",
* closeOnSubmit: false,
* submitOnClose: true,
* submitOnChange: true,
* resizable: true,
* baseApplication: "ActorSheet",
* dragDrop: [{dragSelector: ".item-list .item", dropSelector: null}]
* })
* ```
*/
static get defaultOptions(): BaseEntitySheet.Options;
/* ... */
}
The type should most likely be ConstructorOf<NameOfTheClass>
. This will also allow deriving classes to be used as value. In rare occasions (i.e. when really only instances of this specific class may be assigned, no deriving classes), typeof NameOfTheClass
can be used.
Example:
/**
* Configuration for the ActiveEffect embedded Entity
*/
ActiveEffect: {
/**
* @defaultValue `ActiveEffect`
*/
entityClass: ConstructorOf<ActiveEffect>;
/**
* @defaultValue `ActiveEffectConfig`
*/
sheetClass: ConstructorOf<ActiveEffectConfig>;
};
This is just a static property of the class. Add it to the class at the very bottom.
Example:
In foundry.js
class AVSettings {
/* ... */
}
AVSettings.AV_MODES = {
DISABLED: 0,
AUDIO: 1,
VIDEO: 2,
AUDIO_VIDEO: 3
};
Type definition
declare class AVSettings {
/* ... */
static AV_MODES: {
DISABLED: 0;
AUDIO: 1;
VIDEO: 2;
AUDIO_VIDEO: 3;
};
}
If you've made it this far: Thanks for reading and for trying to contribute. It is much appreciated. We're hoping to see you soon on the League Discord!