diff --git a/changelog.md b/changelog.md
index 786bea1c..928e4eb5 100644
--- a/changelog.md
+++ b/changelog.md
@@ -4,7 +4,6 @@
- Added API endpoints:
- `ItemPiles.API.getItemPileItemTypeFilters(TokenDocument|Actor)` - Returns the item type filters for a given item pile
- `ItemPiles.API.getItemPileItems(TokenDocument|Actor, Array|Boolean)` - Returns the items the item pile contains and can transfer
-- Added warning on startup for module incompatibilities
- Updated japanese localization
- Fixed item piles not respecting item type filters
- Fixed issue with `ItemPiles.API.turnTokenIntoItemPile` not actually turning the token into an item pile
diff --git a/docs/api.md b/docs/api.md
index 8f4447ba..16babfe4 100644
--- a/docs/api.md
+++ b/docs/api.md
@@ -20,7 +20,7 @@
## Functions
-### Settings
+### Setting Methods
setActorClassType(inClassType) ⇒ Promise
@@ -39,7 +39,7 @@
Sets the filters for item types eligible for interaction within this system
-### Item Piles
+### Item Pile Methods
createItemPile(position, [items], [pileActorName]) ⇒ Promise
Creates the default item pile token at a location.
@@ -108,28 +108,28 @@
Causes all connected users to re-render a specific pile's inventory UI
-### Items & Attributes
+### Item and Attribute Methods
-transferItems(source, target, items, [itemTypeFilters]) ⇒ Promise.<object>
-Transfers items from the source to the target, subtracting a number of quantity from the source's item and adding it to the target's item, deleting items from the source if their quantity reaches 0
+addItems(target, items, [itemTypeFilters]) ⇒ Promise.<array>
+Adds item to an actor, increasing item quantities if matches were found
removeItems(target, items, [itemTypeFilters]) ⇒ Promise.<array>
Subtracts the quantity of items on an actor. If the quantity of an item reaches 0, the item is removed from the actor.
-addItems(target, items, [itemTypeFilters]) ⇒ Promise.<array>
-Adds item to an actor, increasing item quantities if matches were found
+transferItems(source, target, items, [itemTypeFilters]) ⇒ Promise.<object>
+Transfers items from the source to the target, subtracting a number of quantity from the source's item and adding it to the target's item, deleting items from the source if their quantity reaches 0
transferAllItems(source, target, [itemTypeFilters]) ⇒ Promise.<array>
Transfers all items between the source and the target.
-transferAttributes(source, target, attributes) ⇒ Promise.<object>
-Transfers a set quantity of an attribute from a source to a target, removing it or subtracting from the source and adds it the target
+addAttributes(target, attributes) ⇒ Promise.<object>
+Adds to attributes on an actor
removeAttributes(target, attributes) ⇒ Promise.<object>
Subtracts attributes on the target
-addAttributes(target, attributes) ⇒ Promise.<object>
-Adds to attributes on an actor
+transferAttributes(source, target, attributes) ⇒ Promise.<object>
+Transfers a set quantity of an attribute from a source to a target, removing it or subtracting from the source and adds it the target
transferAllAttributes(source, target) ⇒ Promise.<object>
Transfers all dynamic attributes from a source to a target, removing it or subtracting from the source and adding them to the target
@@ -138,7 +138,7 @@
Transfers all items and attributes between the source and the target.
-### Utility
+### Utility Methods
rerenderTokenHud() ⇒ Promise
Causes every user's token HUD to rerender
@@ -151,196 +151,97 @@ or not, returning the type if it is NOT allowed.
-## ItemPiles.API.ACTOR\_CLASS\_TYPE ⇒ string
+## ACTOR\_CLASS\_TYPE ⇒ string
The actor class type used for the original item pile actor in this system
-## ItemPiles.API.DYNAMIC\_ATTRIBUTES ⇒ array
+## DYNAMIC\_ATTRIBUTES ⇒ array
The attributes used to track dynamic attributes in this system
-## ItemPiles.API.ITEM\_QUANTITY\_ATTRIBUTE ⇒ string
+## ITEM\_QUANTITY\_ATTRIBUTE ⇒ string
The attribute used to track the quantity of items in this system
-## ItemPiles.API.ITEM\_TYPE\_ATTRIBUTE ⇒ string
+## ITEM\_TYPE\_ATTRIBUTE ⇒ string
The attribute used to track the item type in this system
-## ItemPiles.API.ITEM\_TYPE\_FILTERS ⇒ Array
+## ITEM\_TYPE\_FILTERS ⇒ Array
The filters for item types eligible for interaction within this system
-## ItemPiles.API.setActorClassType(inClassType) ⇒ Promise
+## setActorClassType(inClassType) ⇒ Promise
Sets the actor class type used for the original item pile actor in this system
+
| Param | Type |
| --- | --- |
| inClassType | string
|
-## ItemPiles.API.setDynamicAttributes(inAttributes) ⇒ Promise
+## setDynamicAttributes(inAttributes) ⇒ Promise
Sets the attributes used to track dynamic attributes in this system
+
| Param | Type |
| --- | --- |
| inAttributes | array
|
-## ItemPiles.API.setItemQuantityAttribute(inAttribute) ⇒ Promise
+## setItemQuantityAttribute(inAttribute) ⇒ Promise
Sets the inAttribute used to track the quantity of items in this system
+
| Param | Type |
| --- | --- |
| inAttribute | string
|
-## ItemPiles.API.setItemTypeAttribute(inAttribute) ⇒ string
+## setItemTypeAttribute(inAttribute) ⇒ string
Sets the attribute used to track the item type in this system
+
| Param | Type |
| --- | --- |
| inAttribute | string
|
-## ItemPiles.API.setItemTypeFilters(inFilters) ⇒ Promise
+## setItemTypeFilters(inFilters) ⇒ Promise
Sets the filters for item types eligible for interaction within this system
+
| Param | Type |
| --- | --- |
| inFilters | string/array
|
-## ItemPiles.API.createItemPile(position, [items], [pileActorName]) ⇒ Promise
+## createItemPile(position, [items], [pileActorName]) ⇒ Promise
Creates the default item pile token at a location.
+
| Param | Type | Default | Description |
| --- | --- | --- | --- |
| position | object
| | The position to create the item pile at |
| [items] | array/boolean
| false
| Any items to create on the item pile |
| [pileActorName] | string/boolean
| false
| Whether to use an existing item pile actor as the basis of this new token |
-
-
-## ItemPiles.API.transferItems(source, target, items, [itemTypeFilters]) ⇒ Promise.<object>
-Transfers items from the source to the target, subtracting a number of quantity from the source's item and adding it to the target's item, deleting items from the source if their quantity reaches 0
-**Returns**: Promise.<object>
- An object containing a key value pair for each item added to the target, key being item ID, value being quantities added
-
-| Param | Type | Default | Description |
-| --- | --- | --- | --- |
-| source | Actor/Token/TokenDocument
| | The source to transfer the items from |
-| target | Actor/Token/TokenDocument
| | The target to transfer the items to |
-| items | array
| | An array of objects each containing the item id (key "_id") and the quantity to transfer (key "quantity") |
-| [itemTypeFilters] | array/boolean
| false
| Array of item types disallowed - will default to module settings if none provided |
-
-
-
-## ItemPiles.API.removeItems(target, items, [itemTypeFilters]) ⇒ Promise.<array>
-Subtracts the quantity of items on an actor. If the quantity of an item reaches 0, the item is removed from the actor.
-**Returns**: Promise.<array>
- An array containing the objects of each item that was removed, with their quantities set to the number removed
-
-| Param | Type | Default | Description |
-| --- | --- | --- | --- |
-| target | Actor/Token/TokenDocument
| | The target to remove a items from |
-| items | array
| | An array of objects each containing the item id (key "_id") and the quantity to remove (key "quantity"), or an array of IDs to remove all quantities of |
-| [itemTypeFilters] | array/boolean
| false
| Array of item types disallowed - will default to module settings if none provided |
-
-
-
-## ItemPiles.API.addItems(target, items, [itemTypeFilters]) ⇒ Promise.<array>
-Adds item to an actor, increasing item quantities if matches were found
-**Returns**: Promise.<array>
- An array containing each item added as an object, with their quantities updated to match the new amounts
-
-| Param | Type | Default | Description |
-| --- | --- | --- | --- |
-| target | Actor/TokenDocument/Token
| | The target to add an item to |
-| items | array
| | An array of item objects |
-| [itemTypeFilters] | array/boolean
| false
| Array of item types disallowed - will default to module settings if none provided |
-
-
-
-## ItemPiles.API.transferAllItems(source, target, [itemTypeFilters]) ⇒ Promise.<array>
-Transfers all items between the source and the target.
-**Returns**: Promise.<array>
- An array containing all of the items that were transferred to the target
-
-| Param | Type | Default | Description |
-| --- | --- | --- | --- |
-| source | Actor/Token/TokenDocument
| | The actor to transfer all items from |
-| target | Actor/Token/TokenDocument
| | The actor to receive all the items |
-| [itemTypeFilters] | array/boolean
| false
| Array of item types disallowed - will default to module settings if none provided |
-
-
-
-## ItemPiles.API.transferAttributes(source, target, attributes) ⇒ Promise.<object>
-Transfers a set quantity of an attribute from a source to a target, removing it or subtracting from the source and adds it the target
-**Returns**: Promise.<object>
- An object containing a key value pair of each attribute transferred, the key being the attribute path and its value being the quantity that was transferred
-
-| Param | Type | Description |
-| --- | --- | --- |
-| source | Actor/Token/TokenDocument
| The source to transfer the attribute from |
-| target | Actor/Token/TokenDocument
| The target to transfer the attribute to |
-| attributes | array/object
| This can be either an array of attributes to transfer (to transfer all of a given attribute), or an object with each key being an attribute path, and its value being the quantity to transfer |
-
-
-
-## ItemPiles.API.removeAttributes(target, attributes) ⇒ Promise.<object>
-Subtracts attributes on the target
-**Returns**: Promise.<object>
- Returns an array containing a key value pair of the attribute path and the quantity of that attribute that was removed
-
-| Param | Type | Description |
-| --- | --- | --- |
-| target | Token/TokenDocument
| The target whose attributes will be subtracted from |
-| attributes | array/object
| This can be either an array of attributes to subtract (to zero out a given attribute), or an object with each key being an attribute path, and its value being the quantity to subtract |
-
-
-
-## ItemPiles.API.addAttributes(target, attributes) ⇒ Promise.<object>
-Adds to attributes on an actor
-**Returns**: Promise.<object>
- Returns an array containing a key value pair of the attribute path and the quantity of that attribute that was removed
-
-| Param | Type | Description |
-| --- | --- | --- |
-| target | Actor/Token/TokenDocument
| The target whose attribute will have a set quantity added to it |
-| attributes | object
| An object with each key being an attribute path, and its value being the quantity to add |
-
-
-
-## ItemPiles.API.transferAllAttributes(source, target) ⇒ Promise.<object>
-Transfers all dynamic attributes from a source to a target, removing it or subtracting from the source and adding them to the target
-**Returns**: Promise.<object>
- An object containing a key value pair of each attribute transferred, the key being the attribute path and its value being the quantity that was transferred
-
-| Param | Type | Description |
-| --- | --- | --- |
-| source | Actor/Token/TokenDocument
| The source to transfer the attributes from |
-| target | Actor/Token/TokenDocument
| The target to transfer the attributes to |
-
-
-
-## ItemPiles.API.transferEverything(source, target, [itemTypeFilters]) ⇒ Promise.<object>
-Transfers all items and attributes between the source and the target.
-**Returns**: Promise.<object>
- An object containing all items and attributes transferred to the target
-
-| Param | Type | Default | Description |
-| --- | --- | --- | --- |
-| source | Actor/Token/TokenDocument
| | The actor to transfer all items and attributes from |
-| target | Actor/Token/TokenDocument
| | The actor to receive all the items and attributes |
-| [itemTypeFilters] | array/boolean
| false
| Array of item types disallowed - will default to module settings if none provided |
-
-## ItemPiles.API.turnTokenIntoItemPile(target, pileSettings, tokenSettings) ⇒ Promise.<string>
+## turnTokenIntoItemPile(target, pileSettings, tokenSettings) ⇒ Promise.<string>
Turns a token and its actor into an item pile
-**Returns**: Promise.<string>
- The uuid of the target after it was turned into an item pile
+
+**Returns**: Promise.<string>
- The uuid of the target after it was turned into an item pile
| Param | Type | Description |
| --- | --- | --- |
@@ -350,24 +251,22 @@ Turns a token and its actor into an item pile
-## ItemPiles.API.revertTokenFromItemPile(target, tokenSettings) ⇒ Promise.<string>
+## revertTokenFromItemPile(target, tokenSettings) ⇒ Promise.<string>
Reverts a token from an item pile into a normal token and actor
-**Returns**: Promise.<string>
- The uuid of the target after it was reverted from an item pile
+
+**Returns**: Promise.<string>
- The uuid of the target after it was reverted from an item pile
| Param | Type | Description |
| --- | --- | --- |
| target | Token/TokenDocument
| The target to be reverted from an item pile |
| tokenSettings | object
| Overriding settings that will update the token |
-
-
-## ItemPiles.API.rerenderTokenHud() ⇒ Promise
-Causes every user's token HUD to rerender
-## ItemPiles.API.openItemPile(target, [interactingToken]) ⇒ Promise
+## openItemPile(target, [interactingToken]) ⇒ Promise
Opens a pile if it is enabled and a container
+
| Param | Type | Default |
| --- | --- | --- |
| target | Token/TokenDocument
| |
@@ -375,9 +274,10 @@ Opens a pile if it is enabled and a container
-## ItemPiles.API.closeItemPile(target, [interactingToken]) ⇒ Promise
+## closeItemPile(target, [interactingToken]) ⇒ Promise
Closes a pile if it is enabled and a container
+
| Param | Type | Default | Description |
| --- | --- | --- | --- |
| target | Token/TokenDocument
| | Target pile to close |
@@ -385,9 +285,10 @@ Closes a pile if it is enabled and a container
-## ItemPiles.API.toggleItemPileClosed(target, [interactingToken]) ⇒ Promise
+## toggleItemPileClosed(target, [interactingToken]) ⇒ Promise
Toggles a pile's closed state if it is enabled and a container
+
| Param | Type | Default | Description |
| --- | --- | --- | --- |
| target | Token/TokenDocument
| | Target pile to open or close |
@@ -395,9 +296,10 @@ Toggles a pile's closed state if it is enabled and a container
-## ItemPiles.API.lockItemPile(target, [interactingToken]) ⇒ Promise
+## lockItemPile(target, [interactingToken]) ⇒ Promise
Locks a pile if it is enabled and a container
+
| Param | Type | Default | Description |
| --- | --- | --- | --- |
| target | Token/TokenDocument
| | Target pile to lock |
@@ -405,9 +307,10 @@ Locks a pile if it is enabled and a container
-## ItemPiles.API.unlockItemPile(target, [interactingToken]) ⇒ Promise
+## unlockItemPile(target, [interactingToken]) ⇒ Promise
Unlocks a pile if it is enabled and a container
+
| Param | Type | Default | Description |
| --- | --- | --- | --- |
| target | Token/TokenDocument
| | Target pile to unlock |
@@ -415,9 +318,10 @@ Unlocks a pile if it is enabled and a container
-## ItemPiles.API.toggleItemPileLocked(target, [interactingToken]) ⇒ Promise
+## toggleItemPileLocked(target, [interactingToken]) ⇒ Promise
Toggles a pile's locked state if it is enabled and a container
+
| Param | Type | Default | Description |
| --- | --- | --- | --- |
| target | Token/TokenDocument
| | Target pile to lock or unlock |
@@ -425,45 +329,50 @@ Toggles a pile's locked state if it is enabled and a container
-## ItemPiles.API.rattleItemPile(target) ⇒ Promise.<boolean>
+## rattleItemPile(target) ⇒ Promise.<boolean>
Causes the item pile to play a sound as it was attempted to be opened, but was locked
+
| Param | Type |
| --- | --- |
| target | Token/TokenDocument
|
-## ItemPiles.API.isItemPileLocked(target) ⇒ boolean
+## isItemPileLocked(target) ⇒ boolean
Whether an item pile is locked. If it is not enabled or not a container, it is always false.
+
| Param | Type |
| --- | --- |
| target | Token/TokenDocument
|
-## ItemPiles.API.isItemPileClosed(target) ⇒ boolean
+## isItemPileClosed(target) ⇒ boolean
Whether an item pile is closed. If it is not enabled or not a container, it is always false.
+
| Param | Type |
| --- | --- |
| target | Token/TokenDocument
|
-## ItemPiles.API.isItemPileContainer(target) ⇒ boolean
+## isItemPileContainer(target) ⇒ boolean
Whether an item pile is a container. If it is not enabled, it is always false.
+
| Param | Type |
| --- | --- |
| target | Token/TokenDocument
|
-## ItemPiles.API.updateItemPile(target, newData, [interactingToken], [tokenSettings]) ⇒ Promise
+## updateItemPile(target, newData, [interactingToken], [tokenSettings]) ⇒ Promise
Updates a pile with new data.
+
| Param | Type | Default |
| --- | --- | --- |
| target | Token/TokenDocument
| |
@@ -473,56 +382,50 @@ Updates a pile with new data.
-## ItemPiles.API.deleteItemPile(target) ⇒ Promise
+## deleteItemPile(target) ⇒ Promise
Deletes a pile, calling the relevant hooks.
+
| Param | Type |
| --- | --- |
| target | Token/TokenDocument
|
-
-
-## ItemPiles.API.isItemTypeDisallowed(item, [itemTypeFilters]) ⇒ boolean/string
-Checks whether an item (or item data) is of a type that is not allowed. If an array whether that type is allowed
-or not, returning the type if it is NOT allowed.
-
-| Param | Type | Default |
-| --- | --- | --- |
-| item | Item/Object
| |
-| [itemTypeFilters] | array/boolean
| false
|
-
-## ItemPiles.API.isValidItemPile(document) ⇒ boolean
+## isValidItemPile(document) ⇒ boolean
Whether a given document is a valid pile or not
+
| Param | Type |
| --- | --- |
| document | TokenDocument
\| Actor
|
-## ItemPiles.API.isItemPileEmpty(target) ⇒ boolean
+## isItemPileEmpty(target) ⇒ boolean
Whether the item pile is empty
+
| Param | Type |
| --- | --- |
| target | TokenDocument
\| Actor
|
-## ItemPiles.API.getItemPileItemTypeFilters(target) ⇒ Array
+## getItemPileItemTypeFilters(target) ⇒ Array
Returns the item type filters for a given item pile
+
| Param |
| --- |
| target |
-## ItemPiles.API.getItemPileItems(target, [itemTypeFilters]) ⇒ Array
+## getItemPileItems(target, [itemTypeFilters]) ⇒ Array
Returns the items this item pile can transfer
+
| Param | Type | Default | Description |
| --- | --- | --- | --- |
| target | TokenDocument
\| Actor
| | |
@@ -530,29 +433,164 @@ Returns the items this item pile can transfer
-## ItemPiles.API.getItemPileAttributes(target) ⇒ array
+## getItemPileAttributes(target) ⇒ array
Returns the attributes this item pile can transfer
+
| Param | Type |
| --- | --- |
| target | TokenDocument
\| Actor
|
-## ItemPiles.API.refreshItemPile(target) ⇒ Promise
+## refreshItemPile(target) ⇒ Promise
Refreshes the target image of an item pile, ensuring it remains in sync
+
| Param |
| --- |
| target |
-## ItemPiles.API.rerenderItemPileInventoryApplication(inPileUuid, [deleted]) ⇒ Promise
+## rerenderItemPileInventoryApplication(inPileUuid, [deleted]) ⇒ Promise
Causes all connected users to re-render a specific pile's inventory UI
+
| Param | Type | Default | Description |
| --- | --- | --- | --- |
| inPileUuid | string
| | The uuid of the pile to be re-rendered |
| [deleted] | boolean
| false
| Whether the pile was deleted as a part of this re-render |
+
+
+## addItems(target, items, [itemTypeFilters]) ⇒ Promise.<array>
+Adds item to an actor, increasing item quantities if matches were found
+
+**Returns**: Promise.<array>
- An array containing each item added as an object, with their quantities updated to match the new amounts
+
+| Param | Type | Default | Description |
+| --- | --- | --- | --- |
+| target | Actor/TokenDocument/Token
| | The target to add an item to |
+| items | array
| | An array of item objects |
+| [itemTypeFilters] | array/boolean
| false
| Array of item types disallowed - will default to module settings if none provided |
+
+
+
+## removeItems(target, items, [itemTypeFilters]) ⇒ Promise.<array>
+Subtracts the quantity of items on an actor. If the quantity of an item reaches 0, the item is removed from the actor.
+
+**Returns**: Promise.<array>
- An array containing the objects of each item that was removed, with their quantities set to the number removed
+
+| Param | Type | Default | Description |
+| --- | --- | --- | --- |
+| target | Actor/Token/TokenDocument
| | The target to remove a items from |
+| items | array
| | An array of objects each containing the item id (key "_id") and the quantity to remove (key "quantity"), or an array of IDs to remove all quantities of |
+| [itemTypeFilters] | array/boolean
| false
| Array of item types disallowed - will default to module settings if none provided |
+
+
+
+## transferItems(source, target, items, [itemTypeFilters]) ⇒ Promise.<object>
+Transfers items from the source to the target, subtracting a number of quantity from the source's item and adding it to the target's item, deleting items from the source if their quantity reaches 0
+
+**Returns**: Promise.<object>
- An object containing a key value pair for each item added to the target, key being item ID, value being quantities added
+
+| Param | Type | Default | Description |
+| --- | --- | --- | --- |
+| source | Actor/Token/TokenDocument
| | The source to transfer the items from |
+| target | Actor/Token/TokenDocument
| | The target to transfer the items to |
+| items | array
| | An array of objects each containing the item id (key "_id") and the quantity to transfer (key "quantity") |
+| [itemTypeFilters] | array/boolean
| false
| Array of item types disallowed - will default to module settings if none provided |
+
+
+
+## transferAllItems(source, target, [itemTypeFilters]) ⇒ Promise.<array>
+Transfers all items between the source and the target.
+
+**Returns**: Promise.<array>
- An array containing all of the items that were transferred to the target
+
+| Param | Type | Default | Description |
+| --- | --- | --- | --- |
+| source | Actor/Token/TokenDocument
| | The actor to transfer all items from |
+| target | Actor/Token/TokenDocument
| | The actor to receive all the items |
+| [itemTypeFilters] | array/boolean
| false
| Array of item types disallowed - will default to module settings if none provided |
+
+
+
+## addAttributes(target, attributes) ⇒ Promise.<object>
+Adds to attributes on an actor
+
+**Returns**: Promise.<object>
- Returns an array containing a key value pair of the attribute path and the quantity of that attribute that was removed
+
+| Param | Type | Description |
+| --- | --- | --- |
+| target | Actor/Token/TokenDocument
| The target whose attribute will have a set quantity added to it |
+| attributes | object
| An object with each key being an attribute path, and its value being the quantity to add |
+
+
+
+## removeAttributes(target, attributes) ⇒ Promise.<object>
+Subtracts attributes on the target
+
+**Returns**: Promise.<object>
- Returns an array containing a key value pair of the attribute path and the quantity of that attribute that was removed
+
+| Param | Type | Description |
+| --- | --- | --- |
+| target | Token/TokenDocument
| The target whose attributes will be subtracted from |
+| attributes | array/object
| This can be either an array of attributes to subtract (to zero out a given attribute), or an object with each key being an attribute path, and its value being the quantity to subtract |
+
+
+
+## transferAttributes(source, target, attributes) ⇒ Promise.<object>
+Transfers a set quantity of an attribute from a source to a target, removing it or subtracting from the source and adds it the target
+
+**Returns**: Promise.<object>
- An object containing a key value pair of each attribute transferred, the key being the attribute path and its value being the quantity that was transferred
+
+| Param | Type | Description |
+| --- | --- | --- |
+| source | Actor/Token/TokenDocument
| The source to transfer the attribute from |
+| target | Actor/Token/TokenDocument
| The target to transfer the attribute to |
+| attributes | array/object
| This can be either an array of attributes to transfer (to transfer all of a given attribute), or an object with each key being an attribute path, and its value being the quantity to transfer |
+
+
+
+## transferAllAttributes(source, target) ⇒ Promise.<object>
+Transfers all dynamic attributes from a source to a target, removing it or subtracting from the source and adding them to the target
+
+**Returns**: Promise.<object>
- An object containing a key value pair of each attribute transferred, the key being the attribute path and its value being the quantity that was transferred
+
+| Param | Type | Description |
+| --- | --- | --- |
+| source | Actor/Token/TokenDocument
| The source to transfer the attributes from |
+| target | Actor/Token/TokenDocument
| The target to transfer the attributes to |
+
+
+
+## transferEverything(source, target, [itemTypeFilters]) ⇒ Promise.<object>
+Transfers all items and attributes between the source and the target.
+
+**Returns**: Promise.<object>
- An object containing all items and attributes transferred to the target
+
+| Param | Type | Default | Description |
+| --- | --- | --- | --- |
+| source | Actor/Token/TokenDocument
| | The actor to transfer all items and attributes from |
+| target | Actor/Token/TokenDocument
| | The actor to receive all the items and attributes |
+| [itemTypeFilters] | array/boolean
| false
| Array of item types disallowed - will default to module settings if none provided |
+
+
+
+## rerenderTokenHud() ⇒ Promise
+Causes every user's token HUD to rerender
+
+
+
+## isItemTypeDisallowed(item, [itemTypeFilters]) ⇒ boolean/string
+Checks whether an item (or item data) is of a type that is not allowed. If an array whether that type is allowed
+or not, returning the type if it is NOT allowed.
+
+
+| Param | Type | Default |
+| --- | --- | --- |
+| item | Item/Object
| |
+| [itemTypeFilters] | array/boolean
| false
|
+
diff --git a/scripts/api.js b/scripts/api.js
index bb6075de..3f764e53 100644
--- a/scripts/api.js
+++ b/scripts/api.js
@@ -257,720 +257,822 @@ export default class API {
}
/**
- * Transfers items from the source to the target, subtracting a number of quantity from the source's item and adding it to the target's item, deleting items from the source if their quantity reaches 0
+ * Turns a token and its actor into an item pile
*
- * @param {Actor/Token/TokenDocument} source The source to transfer the items from
- * @param {Actor/Token/TokenDocument} target The target to transfer the items to
- * @param {array} items An array of objects each containing the item id (key "_id") and the quantity to transfer (key "quantity")
- * @param {array/boolean} [itemTypeFilters=false] Array of item types disallowed - will default to module settings if none provided
+ * @param {Token/TokenDocument} target The target to be turned into an item pile
+ * @param {object} pileSettings Overriding settings to be put on the item pile's settings
+ * @param {object} tokenSettings Overriding settings that will update the token
*
- * @returns {Promise} An object containing a key value pair for each item added to the target, key being item ID, value being quantities added
+ * @return {Promise} The uuid of the target after it was turned into an item pile
*/
- static async transferItems(source, target, items, { itemTypeFilters = false } = {}) {
-
- const hookResult = Hooks.call(HOOKS.ITEM.PRE_TRANSFER, source, target, items);
+ static async turnTokenIntoItemPile(target, { pileSettings = {}, tokenSettings = {} } = {}) {
+ const hookResult = Hooks.call(HOOKS.PILE.PRE_TURN_INTO, target, pileSettings, tokenSettings);
if (hookResult === false) return;
-
- const sourceUuid = lib.getUuid(source);
- if (!sourceUuid) throw lib.custom_error(`TransferItems | Could not determine the UUID, please provide a valid source`, true)
-
- if (itemTypeFilters) {
- itemTypeFilters.forEach(filter => {
- if (typeof filter !== "string") throw lib.custom_error(`TransferItems | entries in the itemTypeFilters must be of type string`);
- })
- }
-
- const sourceActorItems = API.getItemPileItems(source);
-
- items.forEach(item => {
- const actorItem = sourceActorItems.find(actorItem => actorItem.id === item._id);
- if (!actorItem) {
- throw lib.custom_error(`TransferItems | Could not find item with id "${item._id}" on source "${sourceUuid}"`, true)
- }
- const disallowedType = API.isItemTypeDisallowed(actorItem, itemTypeFilters);
- if (disallowedType) {
- throw lib.custom_error(`TransferItems | Could not transfer item of type "${disallowedType}"`, true)
- }
- });
-
const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`TransferItems | Could not determine the UUID, please provide a valid target`, true)
-
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.TRANSFER_ITEMS, sourceUuid, targetUuid, items, { itemTypeFilters });
-
+ if (!targetUuid) throw lib.custom_error(`TurnIntoItemPile | Could not determine the UUID, please provide a valid target`, true)
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.TURN_INTO_PILE, targetUuid, pileSettings, tokenSettings);
}
/**
* @private
*/
- static async _transferItems(sourceUuid, targetUuid, items, { itemTypeFilters = false, isEverything = false } = {}) {
-
- const itemsRemoved = await API._removeItems(sourceUuid, items, { itemTypeFilters, isTransfer: true });
-
- const itemsAdded = await API._addItems(targetUuid, itemsRemoved, { itemTypeFilters, isTransfer: true });
-
- if (!isEverything) {
+ static async _turnTokenIntoItemPile(targetUuid, pileSettings = {}, tokenSettings = {}) {
- await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ITEM.TRANSFER, sourceUuid, targetUuid, itemsAdded);
+ const target = await fromUuid(targetUuid);
- const macroData = {
- action: "transferItems",
- source: sourceUuid,
- target: targetUuid,
- itemsAdded: itemsAdded
- };
- await API._executeItemPileMacro(sourceUuid, macroData);
- await API._executeItemPileMacro(targetUuid, macroData);
+ const existingPileSettings = foundry.utils.mergeObject(CONSTANTS.PILE_DEFAULTS, lib.getItemPileData(target));
+ const newPileSettings = foundry.utils.mergeObject(existingPileSettings, pileSettings);
+ newPileSettings.enabled = true;
- const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(sourceUuid);
- await API.rerenderItemPileInventoryApplication(sourceUuid, shouldBeDeleted);
- await API.rerenderItemPileInventoryApplication(targetUuid);
+ await API._updateItemPile(targetUuid, newPileSettings, { tokenSettings });
- if (shouldBeDeleted) {
- await API._deleteItemPile(sourceUuid);
- }
+ setTimeout(API.rerenderTokenHud, 100);
- }
+ await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.PILE.TURN_INTO, targetUuid, newPileSettings);
- return itemsAdded;
+ return targetUuid;
}
/**
- * Subtracts the quantity of items on an actor. If the quantity of an item reaches 0, the item is removed from the actor.
+ * Reverts a token from an item pile into a normal token and actor
*
- * @param {Actor/Token/TokenDocument} target The target to remove a items from
- * @param {array} items An array of objects each containing the item id (key "_id") and the quantity to remove (key "quantity"), or an array of IDs to remove all quantities of
- * @param {array/boolean} [itemTypeFilters=false] Array of item types disallowed - will default to module settings if none provided
+ * @param {Token/TokenDocument} target The target to be reverted from an item pile
+ * @param {object} tokenSettings Overriding settings that will update the token
*
- * @returns {Promise} An array containing the objects of each item that was removed, with their quantities set to the number removed
+ * @return {Promise} The uuid of the target after it was reverted from an item pile
*/
- static async removeItems(target, items, { itemTypeFilters = false } = {}) {
-
- const hookResult = Hooks.call(HOOKS.ITEM.PRE_REMOVE, target, items);
+ static async revertTokenFromItemPile(target, { tokenSettings = {} } = {}) {
+ const hookResult = Hooks.call(HOOKS.PILE.PRE_REVERT_FROM, target, tokenSettings);
if (hookResult === false) return;
-
const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`RemoveItems | Could not determine the UUID, please provide a valid target`, true);
-
- if (itemTypeFilters) {
- itemTypeFilters.forEach(filter => {
- if (typeof filter !== "string") throw lib.custom_error(`RemoveItems | entries in the itemTypeFilters must be of type string`);
- })
- }
-
- const targetActorItems = API.getItemPileItems(target);
-
- items.forEach(item => {
- const itemId = typeof item === "string" ? item : item._id;
- const actorItem = targetActorItems.find(actorItem => actorItem.id === itemId);
- if (!actorItem) {
- throw lib.custom_error(`RemoveItems | Could not find item with id "${itemId}" on target "${targetUuid}"`, true)
- }
- const disallowedType = API.isItemTypeDisallowed(actorItem, itemTypeFilters);
- if (disallowedType) {
- throw lib.custom_error(`RemoveItems | Could not transfer item of type "${disallowedType}"`, true)
- }
- });
-
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.REMOVE_ITEMS, targetUuid, items);
+ if (!targetUuid) throw lib.custom_error(`RevertFromItemPile | Could not determine the UUID, please provide a valid target`, true)
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.REVERT_FROM_PILE, targetUuid, tokenSettings);
}
/**
* @private
*/
- static async _removeItems(targetUuid, items, { isTransfer = false } = {}) {
+ static async _revertTokenFromItemPile(targetUuid, tokenSettings) {
const target = await fromUuid(targetUuid);
- const targetActor = target instanceof TokenDocument
- ? target.actor
- : target;
-
- const itemsRemoved = [];
- const itemsToUpdate = [];
- const itemsToDelete = [];
- for (const item of items) {
- const itemId = typeof item === "string" ? item : item._id;
-
- const actorItem = targetActor.items.get(itemId);
- const removedItem = actorItem.toObject();
-
- const currentQuantity = getProperty(actorItem.data, API.ITEM_QUANTITY_ATTRIBUTE);
-
- const quantityToRemove = getProperty(item, API.ITEM_QUANTITY_ATTRIBUTE) ?? item.quantity ?? currentQuantity;
+ const pileSettings = foundry.utils.mergeObject(CONSTANTS.PILE_DEFAULTS, lib.getItemPileData(target));
- const newQuantity = Math.max(0, currentQuantity - quantityToRemove);
+ pileSettings.enabled = false;
- if (newQuantity >= 1) {
- setProperty(removedItem, API.ITEM_QUANTITY_ATTRIBUTE, quantityToRemove);
- itemsToUpdate.push({ _id: actorItem.id, [API.ITEM_QUANTITY_ATTRIBUTE]: newQuantity });
- } else {
- setProperty(removedItem, API.ITEM_QUANTITY_ATTRIBUTE, currentQuantity);
- itemsToDelete.push(actorItem.id);
- }
+ await API._updateItemPile(targetUuid, pileSettings, { tokenSettings });
- itemsRemoved.push(removedItem);
+ setTimeout(API.rerenderTokenHud, 100);
+ if (target instanceof TokenDocument) {
+ await API.rerenderItemPileInventoryApplication(targetUuid);
}
- await targetActor.updateEmbeddedDocuments("Item", itemsToUpdate);
- await targetActor.deleteEmbeddedDocuments("Item", itemsToDelete);
-
- await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ITEM.REMOVE, targetUuid, itemsRemoved);
-
- if (!isTransfer) {
-
- const macroData = {
- action: "removeItems",
- target: targetUuid,
- items: itemsRemoved
- };
-
- await API._executeItemPileMacro(targetUuid, macroData);
-
- const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(targetUuid);
+ await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.PILE.REVERT_FROM, targetUuid);
- await API.rerenderItemPileInventoryApplication(targetUuid, shouldBeDeleted);
+ return targetUuid;
- if (shouldBeDeleted) {
- await API._deleteItemPile(targetUuid);
- }
+ }
+ /**
+ * Opens a pile if it is enabled and a container
+ *
+ * @param {Token/TokenDocument} target
+ * @param {Token/TokenDocument/boolean} [interactingToken=false]
+ *
+ * @return {Promise}
+ */
+ static async openItemPile(target, interactingToken = false) {
+ const data = lib.getItemPileData(target);
+ if (!data?.enabled || !data?.isContainer) return false;
+ const wasLocked = data.locked;
+ data.closed = false;
+ data.locked = false;
+ if (wasLocked) {
+ const hookResult = Hooks.call(HOOKS.PILE.PRE_UNLOCK, target, data, interactingToken);
+ if (hookResult === false) return;
}
-
- return itemsRemoved;
-
+ const hookResult = Hooks.call(HOOKS.PILE.PRE_OPEN, target, data, interactingToken);
+ if (hookResult === false) return;
+ if (data.openSound) {
+ AudioHelper.play({ src: data.openSound })
+ }
+ return API.updateItemPile(target, data, { interactingToken });
}
/**
- * Adds item to an actor, increasing item quantities if matches were found
+ * Closes a pile if it is enabled and a container
*
- * @param {Actor/TokenDocument/Token} target The target to add an item to
- * @param {array} items An array of item objects
- * @param {array/boolean} [itemTypeFilters=false] Array of item types disallowed - will default to module settings if none provided
+ * @param {Token/TokenDocument} target Target pile to close
+ * @param {Token/TokenDocument/boolean} [interactingToken=false]
*
- * @returns {Promise} An array containing each item added as an object, with their quantities updated to match the new amounts
+ * @return {Promise}
*/
- static async addItems(target, items, { itemTypeFilters = false } = {}) {
-
- const hookResult = Hooks.call(HOOKS.ITEM.PRE_ADD, target, items);
+ static async closeItemPile(target, interactingToken = false) {
+ const data = lib.getItemPileData(target);
+ if (!data?.enabled || !data?.isContainer) return false;
+ data.closed = true;
+ const hookResult = Hooks.call(HOOKS.PILE.PRE_CLOSE, target, data, interactingToken);
if (hookResult === false) return;
-
- const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`AddItems | Could not determine the UUID, please provide a valid target`, true)
-
- if (itemTypeFilters) {
- itemTypeFilters.forEach(filter => {
- if (typeof filter !== "string") throw lib.custom_error(`AddItem | entries in the itemTypeFilters must be of type string`);
- })
+ if (data.closeSound) {
+ AudioHelper.play({ src: data.closeSound })
}
+ return API.updateItemPile(target, data, interactingToken);
+ }
- for (const index in items) {
- const item = items[index];
- if (item instanceof Item) {
- items[index] = item.toObject();
- }
- const disallowedType = API.isItemTypeDisallowed(item, itemTypeFilters);
- if (disallowedType) {
- throw lib.custom_error(`AddItems | Could not add item of type "${disallowedType}"`, true)
- }
+ /**
+ * Toggles a pile's closed state if it is enabled and a container
+ *
+ * @param {Token/TokenDocument} target Target pile to open or close
+ * @param {Token/TokenDocument/boolean} [interactingToken=false]
+ *
+ * @return {Promise}
+ */
+ static async toggleItemPileClosed(target, interactingToken = false) {
+ const data = lib.getItemPileData(target);
+ if (!data?.enabled || !data?.isContainer) return false;
+ if (data.closed) {
+ await API.openItemPile(target, interactingToken);
+ } else {
+ await API.closeItemPile(target, interactingToken);
}
-
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.ADD_ITEMS, targetUuid, items);
+ return !data.closed;
}
/**
- * @private
+ * Locks a pile if it is enabled and a container
+ *
+ * @param {Token/TokenDocument} target Target pile to lock
+ * @param {Token/TokenDocument/boolean} [interactingToken=false]
+ *
+ * @return {Promise}
*/
- static async _addItems(targetUuid, items, { isTransfer = false } = {}) {
-
- const target = await fromUuid(targetUuid);
- const targetActor = target instanceof TokenDocument
- ? target.actor
- : target;
-
- const targetActorItems = Array.from(targetActor.items);
-
- const itemsAdded = [];
- const itemsToCreate = [];
- const itemsToUpdate = [];
- for (const itemData of items) {
-
- const item = lib.getSimilarItem(targetActorItems, { itemId: itemData._id, itemName: itemData.name, itemType: itemData.type });
-
- const incomingQuantity = getProperty(itemData, API.ITEM_QUANTITY_ATTRIBUTE) ?? 1;
-
- const itemAdded = item ? item.toObject() : foundry.utils.duplicate(itemData);
-
- if (item) {
- const currentQuantity = getProperty(item.data, API.ITEM_QUANTITY_ATTRIBUTE);
- const newQuantity = currentQuantity + incomingQuantity;
- itemsToUpdate.push({
- "_id": item.id,
- [API.ITEM_QUANTITY_ATTRIBUTE]: newQuantity
- });
-
- const itemAdded = item.toObject();
- setProperty(itemAdded, API.ITEM_QUANTITY_ATTRIBUTE, newQuantity)
- itemsAdded.push(itemAdded);
- } else {
- setProperty(itemAdded, API.ITEM_QUANTITY_ATTRIBUTE, incomingQuantity)
- itemsToCreate.push(itemData);
- }
-
+ static async lockItemPile(target, interactingToken = false) {
+ const data = lib.getItemPileData(target);
+ if (!data?.enabled || !data?.isContainer) return false;
+ const wasClosed = data.closed;
+ data.closed = true;
+ data.locked = true;
+ if (!wasClosed) {
+ const hookResult = Hooks.call(HOOKS.PILE.PRE_CLOSE, target, data, interactingToken);
+ if (hookResult === false) return;
}
+ const hookResult = Hooks.call(HOOKS.PILE.PRE_LOCK, target, data, interactingToken);
+ if (hookResult === false) return;
+ if (data.closeSound && !wasClosed) {
+ AudioHelper.play({ src: data.closeSound })
+ }
+ return API.updateItemPile(target, data, interactingToken);
+ }
- const itemsCreated = await targetActor.createEmbeddedDocuments("Item", itemsToCreate);
- await targetActor.updateEmbeddedDocuments("Item", itemsToUpdate);
-
- itemsCreated.forEach(item => itemsAdded.push(item.toObject()));
-
- await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ITEM.ADD, targetUuid, itemsAdded);
-
- if (!isTransfer) {
-
- const macroData = {
- action: "addItems",
- target: targetUuid,
- items: itemsAdded
- };
-
- await API._executeItemPileMacro(targetUuid, macroData);
-
- await API.rerenderItemPileInventoryApplication(targetUuid);
+ /**
+ * Unlocks a pile if it is enabled and a container
+ *
+ * @param {Token/TokenDocument} target Target pile to unlock
+ * @param {Token/TokenDocument/boolean} [interactingToken=false]
+ *
+ * @return {Promise}
+ */
+ static async unlockItemPile(target, interactingToken = false) {
+ const data = lib.getItemPileData(target);
+ if (!data?.enabled || !data?.isContainer) return false;
+ data.locked = false;
+ Hooks.call(HOOKS.PILE.PRE_UNLOCK, target, data, interactingToken);
+ return API.updateItemPile(target, data, interactingToken);
+ }
+ /**
+ * Toggles a pile's locked state if it is enabled and a container
+ *
+ * @param {Token/TokenDocument} target Target pile to lock or unlock
+ * @param {Token/TokenDocument/boolean} [interactingToken=false]
+ *
+ * @return {Promise}
+ */
+ static async toggleItemPileLocked(target, interactingToken = false) {
+ const data = lib.getItemPileData(target);
+ if (!data?.enabled || !data?.isContainer) return false;
+ if (data.locked) {
+ return API.unlockItemPile(target, interactingToken);
}
+ return API.lockItemPile(target, interactingToken);
+ }
- return itemsAdded;
+ /**
+ * Causes the item pile to play a sound as it was attempted to be opened, but was locked
+ *
+ * @param {Token/TokenDocument} target
+ *
+ * @return {Promise}
+ */
+ static async rattleItemPile(target) {
+ const data = lib.getItemPileData(target);
+ if (!data?.enabled || !data?.isContainer) return false;
+ if (data.locked && data.lockedSound) {
+ AudioHelper.play({ src: data.lockedSound })
+ }
+ return true;
+ }
+ /**
+ * Whether an item pile is locked. If it is not enabled or not a container, it is always false.
+ *
+ * @param {Token/TokenDocument} target
+ *
+ * @return {boolean}
+ */
+ static isItemPileLocked(target) {
+ const data = lib.getItemPileData(target);
+ if (!data?.enabled || !data?.isContainer) return false;
+ return data.locked;
}
/**
- * Transfers all items between the source and the target.
+ * Whether an item pile is closed. If it is not enabled or not a container, it is always false.
*
- * @param {Actor/Token/TokenDocument} source The actor to transfer all items from
- * @param {Actor/Token/TokenDocument} target The actor to receive all the items
- * @param {array/boolean} [itemTypeFilters=false] Array of item types disallowed - will default to module settings if none provided
+ * @param {Token/TokenDocument} target
*
- * @returns {Promise} An array containing all of the items that were transferred to the target
+ * @return {boolean}
*/
- static async transferAllItems(source, target, { itemTypeFilters = false } = {}) {
+ static isItemPileClosed(target) {
+ const data = lib.getItemPileData(target);
+ if (!data?.enabled || !data?.isContainer) return false;
+ return data.closed;
+ }
- const hookResult = Hooks.call(HOOKS.ITEM.PRE_TRANSFER_ALL, source, target, itemTypeFilters);
- if (hookResult === false) return;
+ /**
+ * Whether an item pile is a container. If it is not enabled, it is always false.
+ *
+ * @param {Token/TokenDocument} target
+ *
+ * @return {boolean}
+ */
+ static isItemPileContainer(target) {
+ const data = lib.getItemPileData(target);
+ return data?.enabled && data?.isContainer;
+ }
- const sourceUuid = lib.getUuid(source);
- if (!sourceUuid) throw lib.custom_error(`TransferAllItems | Could not determine the UUID, please provide a valid source`, true)
+ /**
+ * Updates a pile with new data.
+ *
+ * @param {Token/TokenDocument} target
+ * @param {object} newData
+ * @param {Token/TokenDocument/boolean} [interactingToken=false]
+ * @param {object/boolean} [tokenSettings=false]
+ *
+ * @return {Promise}
+ */
+ static async updateItemPile(target, newData, { interactingToken = false, tokenSettings = false } = {}) {
const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`TransferAllItems | Could not determine the UUID, please provide a valid target`, true)
+ if (!targetUuid) throw lib.custom_error(`updateItemPile | Could not determine the UUID, please provide a valid target`, true);
- if (itemTypeFilters) {
- itemTypeFilters.forEach(filter => {
- if (typeof filter !== "string") throw lib.custom_error(`RevertFromItemPile | entries in the itemTypeFilters must be of type string`);
- })
- }
+ const interactingTokenUuid = interactingToken ? lib.getUuid(interactingToken) : false;
+ if (interactingToken && !interactingTokenUuid) throw lib.custom_error(`updateItemPile | Could not determine the UUID, please provide a valid target`, true);
- return itemPileSocket.executeAsGM(
- SOCKET_HANDLERS.TRANSFER_ALL_ITEMS,
- sourceUuid,
- targetUuid,
- {
- itemTypeFilters
- }
- );
+ const hookResult = Hooks.call(HOOKS.PILE.PRE_UPDATE, target, newData, interactingToken, tokenSettings);
+ if (hookResult === false) return;
+
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.UPDATE_PILE, targetUuid, newData, {
+ interactingTokenUuid,
+ tokenSettings
+ })
}
/**
* @private
*/
- static async _transferAllItems(sourceUuid, targetUuid, { itemTypeFilters = false, isEverything = false } = {}) {
-
- const source = await fromUuid(sourceUuid);
- if (!source) throw lib.custom_error(`TransferAllItems | Could not find source with UUID "${sourceUuid}"`, true)
+ static async _updateItemPile(targetUuid, newData, { interactingTokenUuid = false, tokenSettings = false } = {}) {
const target = await fromUuid(targetUuid);
- if (!target) throw lib.custom_error(`TransferAllItems | Could not find target with UUID "${targetUuid}"`, true)
- if (!Array.isArray(itemTypeFilters)) {
- itemTypeFilters = API.ITEM_TYPE_FILTERS;
- } else {
- itemTypeFilters.forEach(filter => {
- if (typeof filter !== "string") throw lib.custom_error(`TransferAllItems | entries in the itemTypeFilters must be of type string`);
- })
+ const oldData = lib.getItemPileData(target);
+
+ const data = foundry.utils.mergeObject(
+ foundry.utils.duplicate(oldData),
+ foundry.utils.duplicate(newData)
+ );
+
+ const diff = foundry.utils.diffObject(oldData, data);
+
+ await lib.wait(25);
+
+ await lib.updateItemPile(target, data, tokenSettings);
+
+ if (data.isEnabled && data.isContainer) {
+ if (diff?.closed === true) {
+ await API._executeItemPileMacro(targetUuid, {
+ action: "closeItemPile",
+ source: interactingTokenUuid,
+ target: targetUuid
+ });
+ }
+ if (diff?.locked === true) {
+ await API._executeItemPileMacro(targetUuid, {
+ action: "lockItemPile",
+ source: interactingTokenUuid,
+ target: targetUuid
+ });
+ }
+ if (diff?.locked === false) {
+ await API._executeItemPileMacro(targetUuid, {
+ action: "unlockItemPile",
+ source: interactingTokenUuid,
+ target: targetUuid
+ });
+ }
+ if (diff?.closed === false) {
+ await API._executeItemPileMacro(targetUuid, {
+ action: "openItemPile",
+ source: interactingTokenUuid,
+ target: targetUuid
+ });
+ }
}
- const itemsToRemove = API.getItemPileItems(target, itemTypeFilters).map(item => item.toObject());
+ return itemPileSocket.executeForEveryone(SOCKET_HANDLERS.UPDATED_PILE, targetUuid, diff, interactingTokenUuid);
+ }
- const itemsRemoved = await API._removeItems(sourceUuid, itemsToRemove, { itemTypeFilters, isTransfer: true });
- const itemsAdded = await API._addItems(targetUuid, itemsRemoved, { itemTypeFilters, isTransfer: true });
+ /**
+ * @private
+ */
+ static async _updatedItemPile(targetUuid, diffData, interactingTokenUuid) {
- if (!isEverything) {
+ const target = await lib.getToken(targetUuid);
- await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.TRANSFER_ALL_ITEMS, HOOKS.ITEM.TRANSFER_ALL, sourceUuid, targetUuid, itemsAdded);
+ const interactingToken = interactingTokenUuid ? await fromUuid(interactingTokenUuid) : false;
- const macroData = {
- action: "transferAllItems",
- source: sourceUuid,
- target: targetUuid,
- items: itemsAdded
- };
- await API._executeItemPileMacro(sourceUuid, macroData);
- await API._executeItemPileMacro(targetUuid, macroData);
+ if (foundry.utils.isObjectEmpty(diffData)) return;
- const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(sourceUuid);
- await API.rerenderItemPileInventoryApplication(sourceUuid, shouldBeDeleted);
- await API.rerenderItemPileInventoryApplication(targetUuid);
+ const data = lib.getItemPileData(target);
- if (shouldBeDeleted) {
- await API._deleteItemPile(sourceUuid);
- }
+ Hooks.callAll(HOOKS.PILE.UPDATE, target, diffData, interactingToken)
+ if (data.isEnabled && data.isContainer) {
+ if (diffData?.closed === true) {
+ Hooks.callAll(HOOKS.PILE.CLOSE, target, interactingToken)
+ }
+ if (diffData?.locked === true) {
+ Hooks.callAll(HOOKS.PILE.LOCK, target, interactingToken)
+ }
+ if (diffData?.locked === false) {
+ Hooks.callAll(HOOKS.PILE.UNLOCK, target, interactingToken)
+ }
+ if (diffData?.closed === false) {
+ Hooks.callAll(HOOKS.PILE.OPEN, target, interactingToken)
+ }
}
-
- return itemsAdded;
}
/**
- * Transfers a set quantity of an attribute from a source to a target, removing it or subtracting from the source and adds it the target
+ * Deletes a pile, calling the relevant hooks.
*
- * @param {Actor/Token/TokenDocument} source The source to transfer the attribute from
- * @param {Actor/Token/TokenDocument} target The target to transfer the attribute to
- * @param {array/object} attributes This can be either an array of attributes to transfer (to transfer all of a given attribute), or an object with each key being an attribute path, and its value being the quantity to transfer
+ * @param {Token/TokenDocument} target
*
- * @returns {Promise} An object containing a key value pair of each attribute transferred, the key being the attribute path and its value being the quantity that was transferred
+ * @return {Promise}
*/
- static async transferAttributes(source, target, attributes) {
-
- const hookResult = Hooks.call(HOOKS.ATTRIBUTE.PRE_TRANSFER, source, target, attributes);
- if (hookResult === false) return;
-
- const sourceUuid = lib.getUuid(source);
- if (!sourceUuid) throw lib.custom_error(`TransferAttributes | Could not determine the UUID, please provide a valid source`, true)
-
+ static async deleteItemPile(target) {
+ if (!API.isValidItemPile(target)) {
+ if (!targetUuid) throw lib.custom_error(`deleteItemPile | This is not an item pile, please provide a valid target`, true);
+ }
const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`TransferAttributes | Could not determine the UUID, please provide a valid target`, true)
+ if (!targetUuid) throw lib.custom_error(`deleteItemPile | Could not determine the UUID, please provide a valid target`, true);
+ if (!targetUuid.includes("Token")) {
+ throw lib.custom_error(`deleteItemPile | Please provide a Token or TokenDocument`, true);
+ }
+ const hookResult = Hooks.call(HOOKS.PILE.PRE_DELETE, target);
+ if (hookResult === false) return;
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.DELETE_PILE, targetUuid);
+ }
- const sourceActor = source instanceof TokenDocument
- ? source.actor
- : source;
+ static async _deleteItemPile(targetUuid) {
+ const target = await lib.getToken(targetUuid);
+ return target.delete();
+ }
+ /**
+ * Whether a given document is a valid pile or not
+ *
+ * @param {TokenDocument|Actor} document
+ * @return {boolean}
+ */
+ static isValidItemPile(document) {
+ const documentActor = document instanceof TokenDocument ? document.actor : document;
+ return document && !document.destroyed && documentActor && lib.getItemPileData(document)?.enabled;
+ }
+ /**
+ * Whether the item pile is empty
+ *
+ * @param {TokenDocument|Actor} target
+ * @returns {boolean}
+ */
+ static isItemPileEmpty(target){
const targetActor = target instanceof TokenDocument
? target.actor
: target;
- if (Array.isArray(attributes)) {
- attributes.forEach(attribute => {
- if (typeof attribute !== "string") {
- throw lib.custom_error(`TransferAttributes | Each attribute in the array must be of type string`, true)
- }
- if (!hasProperty(sourceActor.data, attribute)) {
- throw lib.custom_error(`TransferAttributes | Could not find attribute ${attribute} on source's actor with UUID "${targetUuid}"`, true)
- }
- if (!hasProperty(targetActor.data, attribute)) {
- throw lib.custom_error(`TransferAttributes | Could not find attribute ${attribute} on target's actor with UUID "${targetUuid}"`, true)
- }
- });
- } else {
- Object.entries(attributes).forEach(entry => {
- const [attribute, quantity] = entry;
- if (!hasProperty(sourceActor.data, attribute)) {
- throw lib.custom_error(`TransferAttributes | Could not find attribute ${attribute} on source's actor with UUID "${targetUuid}"`, true)
- }
- if (!hasProperty(targetActor.data, attribute)) {
- throw lib.custom_error(`TransferAttributes | Could not find attribute ${attribute} on target's actor with UUID "${targetUuid}"`, true)
- }
- if (!lib.is_real_number(quantity) && quantity > 0) {
- throw lib.custom_error(`TransferAttributes | Attribute "${attribute}" must be of type number and greater than 0`, true)
- }
- });
- }
+ if(!targetActor) return false;
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.TRANSFER_ATTRIBUTES, sourceUuid, targetUuid, attributes);
+ const hasNoItems = API.getItemPileItems(target).length === 0;
+
+ const attributes = API.getItemPileAttributes(target);
+ const hasEmptyAttributes = attributes.find(attribute => {
+ return hasProperty(targetActor.data, attribute.path) && getProperty(targetActor.data, attribute.path) === 0;
+ })
+ return hasNoItems && hasEmptyAttributes;
+ }
+ /**
+ * Returns the item type filters for a given item pile
+ *
+ * @param target
+ * @returns {Array}
+ */
+ static getItemPileItemTypeFilters(target){
+ if(!API.isValidItemPile(target)) return [];
+ const pileData = lib.getItemPileData(target);
+ return pileData.itemTypeFilters
+ ? pileData.itemTypeFilters.split(',').map(str => str.trim().toLowerCase())
+ : API.ITEM_TYPE_FILTERS;
}
/**
- * @private
+ * Returns the items this item pile can transfer
+ *
+ * @param {TokenDocument|Actor} target
+ * @param {array/boolean} [itemTypeFilters=false] Array of item types disallowed - will default to pile settings or module settings if none provided
+ * @returns {Array}
*/
- static async _transferAttributes(sourceUuid, targetUuid, attributes, { isEverything = false } = {}) {
+ static getItemPileItems(target, itemTypeFilters = false){
- const attributesRemoved = await API._removeAttributes(sourceUuid, attributes, { isTransfer: true });
+ if(!API.isValidItemPile(target)) return [];
- const attributesAdded = await API._addAttributes(targetUuid, attributesRemoved, { isTransfer: true });
+ const pileItemFilters = Array.isArray(itemTypeFilters)
+ ? new Set(itemTypeFilters)
+ : new Set(API.getItemPileItemTypeFilters(target));
- if (!isEverything) {
+ const targetActor = target instanceof TokenDocument
+ ? target.actor
+ : target;
- await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ATTRIBUTE.TRANSFER, sourceUuid, targetUuid, attributesAdded);
+ return Array.from(targetActor.items).filter(item => {
+ const itemType = getProperty(item.data, API.ITEM_TYPE_ATTRIBUTE);
+ return !pileItemFilters.has(itemType);
+ })
- const macroData = {
- action: "transferAttributes",
- source: sourceUuid,
- target: targetUuid,
- attributes: attributesAdded
- };
- await API._executeItemPileMacro(sourceUuid, macroData);
- await API._executeItemPileMacro(targetUuid, macroData);
+ }
- const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(sourceUuid);
- await API.rerenderItemPileInventoryApplication(sourceUuid, shouldBeDeleted);
- await API.rerenderItemPileInventoryApplication(targetUuid);
+ /**
+ * Returns the attributes this item pile can transfer
+ *
+ * @param {TokenDocument|Actor} target
+ * @returns {array}
+ */
+ static getItemPileAttributes(target){
+ const pileData = lib.getItemPileData(target);
+ return pileData.overrideAttributes || API.DYNAMIC_ATTRIBUTES;
+ }
- if (shouldBeDeleted) {
- await API._deleteItemPile(sourceUuid);
- }
+ /**
+ * Refreshes the target image of an item pile, ensuring it remains in sync
+ *
+ * @param target
+ * @return {Promise}
+ */
+ static async refreshItemPile(target) {
+ if (!API.isValidItemPile(target)) return;
+ const targetUuid = lib.getUuid(target);
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.REFRESH_PILE, targetUuid)
+ }
+
+ /**
+ * @private
+ */
+ static async _refreshItemPile(targetUuid) {
+ const targetDocument = await fromUuid(targetUuid);
+ if (!API.isValidItemPile(targetDocument)) return;
+
+ let targets = [targetDocument]
+ if (targetDocument instanceof Actor) {
+ targets = Array.from(canvas.tokens.getDocuments()).filter(token => token.actor === targetDocument);
}
- return attributesAdded
+ return Promise.allSettled(targets.map(_target => {
+ return new Promise(async (resolve) => {
+ const uuid = lib.getUuid(_target);
+ const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(uuid);
+ if (!shouldBeDeleted) {
+ await _target.update({
+ "img": API._getItemPileTokenImage(targetDocument),
+ "scale": API._getItemPileTokenScale(targetDocument),
+ })
+ }
+ resolve();
+ })
+ }));
+ }
+ /**
+ * Causes all connected users to re-render a specific pile's inventory UI
+ *
+ * @param {string} inPileUuid The uuid of the pile to be re-rendered
+ * @param {boolean} [deleted=false] Whether the pile was deleted as a part of this re-render
+ * @return {Promise}
+ */
+ static async rerenderItemPileInventoryApplication(inPileUuid, deleted = false) {
+ return itemPileSocket.executeForEveryone(SOCKET_HANDLERS.RERENDER_PILE_INVENTORY, inPileUuid, deleted);
}
/**
- * Subtracts attributes on the target
+ * @private
+ */
+ static async _rerenderItemPileInventoryApplication(inPileUuid, deleted = false) {
+ return ItemPileInventory.rerenderActiveApp(inPileUuid, deleted);
+ }
+
+ /* --- ITEM AND ATTRIBUTE METHODS --- */
+
+ /**
+ * Adds item to an actor, increasing item quantities if matches were found
*
- * @param {Token/TokenDocument} target The target whose attributes will be subtracted from
- * @param {array/object} attributes This can be either an array of attributes to subtract (to zero out a given attribute), or an object with each key being an attribute path, and its value being the quantity to subtract
+ * @param {Actor/TokenDocument/Token} target The target to add an item to
+ * @param {array} items An array of item objects
+ * @param {array/boolean} [itemTypeFilters=false] Array of item types disallowed - will default to module settings if none provided
*
- * @returns {Promise} Returns an array containing a key value pair of the attribute path and the quantity of that attribute that was removed
+ * @returns {Promise} An array containing each item added as an object, with their quantities updated to match the new amounts
*/
- static async removeAttributes(target, attributes) {
+ static async addItems(target, items, { itemTypeFilters = false } = {}) {
- const hookResult = Hooks.call(HOOKS.ATTRIBUTE.PRE_REMOVE, target, attributes);
+ const hookResult = Hooks.call(HOOKS.ITEM.PRE_ADD, target, items);
if (hookResult === false) return;
const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`RemoveAttributes | Could not determine the UUID, please provide a valid target`, true)
-
- const targetActor = target instanceof TokenDocument
- ? target.actor
- : target;
+ if (!targetUuid) throw lib.custom_error(`AddItems | Could not determine the UUID, please provide a valid target`, true)
- if (Array.isArray(attributes)) {
- attributes.forEach(attribute => {
- if (typeof attribute !== "string") {
- throw lib.custom_error(`RemoveAttributes | Each attribute in the array must be of type string`, true)
- }
- if (!hasProperty(targetActor.data, attribute)) {
- throw lib.custom_error(`RemoveAttributes | Could not find attribute ${attribute} on target's actor with UUID "${targetUuid}"`, true)
- }
- });
- } else {
- Object.entries(attributes).forEach(entry => {
- const [attribute, quantity] = entry;
- if (!hasProperty(targetActor.data, attribute)) {
- throw lib.custom_error(`RemoveAttributes | Could not find attribute ${attribute} on target's actor with UUID "${targetUuid}"`, true)
- }
- if (!lib.is_real_number(quantity) && quantity > 0) {
- throw lib.custom_error(`RemoveAttributes | Attribute "${attribute}" must be of type number and greater than 0`, true)
- }
- });
+ if (itemTypeFilters) {
+ itemTypeFilters.forEach(filter => {
+ if (typeof filter !== "string") throw lib.custom_error(`AddItem | entries in the itemTypeFilters must be of type string`);
+ })
}
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.REMOVE_ATTRIBUTES, targetUuid, attributes);
+ for (const index in items) {
+ const item = items[index];
+ if (item instanceof Item) {
+ items[index] = item.toObject();
+ }
+ const disallowedType = API.isItemTypeDisallowed(item, itemTypeFilters);
+ if (disallowedType) {
+ throw lib.custom_error(`AddItems | Could not add item of type "${disallowedType}"`, true)
+ }
+ }
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.ADD_ITEMS, targetUuid, items);
}
/**
* @private
*/
- static async _removeAttributes(targetUuid, attributes, { isTransfer = false } = {}) {
+ static async _addItems(targetUuid, items, { isTransfer = false } = {}) {
const target = await fromUuid(targetUuid);
const targetActor = target instanceof TokenDocument
? target.actor
: target;
- const updates = {};
- const attributesRemoved = {};
+ const targetActorItems = Array.from(targetActor.items);
- if (Array.isArray(attributes)) {
- attributes = Object.fromEntries(attributes.map(attribute => {
- return [attribute, getProperty(targetActor.data, attribute)];
- }))
- }
+ const itemsAdded = [];
+ const itemsToCreate = [];
+ const itemsToUpdate = [];
+ for (const itemData of items) {
- for (const [attribute, quantityToRemove] of Object.entries(attributes)) {
+ const item = lib.getSimilarItem(targetActorItems, { itemId: itemData._id, itemName: itemData.name, itemType: itemData.type });
- const currentQuantity = getProperty(targetActor.data, attribute);
- const newQuantity = Math.max(0, currentQuantity - quantityToRemove);
+ const incomingQuantity = getProperty(itemData, API.ITEM_QUANTITY_ATTRIBUTE) ?? 1;
- updates[attribute] = newQuantity;
+ const itemAdded = item ? item.toObject() : foundry.utils.duplicate(itemData);
+
+ if (item) {
+ const currentQuantity = getProperty(item.data, API.ITEM_QUANTITY_ATTRIBUTE);
+ const newQuantity = currentQuantity + incomingQuantity;
+ itemsToUpdate.push({
+ "_id": item.id,
+ [API.ITEM_QUANTITY_ATTRIBUTE]: newQuantity
+ });
+
+ const itemAdded = item.toObject();
+ setProperty(itemAdded, API.ITEM_QUANTITY_ATTRIBUTE, newQuantity)
+ itemsAdded.push(itemAdded);
+ } else {
+ setProperty(itemAdded, API.ITEM_QUANTITY_ATTRIBUTE, incomingQuantity)
+ itemsToCreate.push(itemData);
+ }
- // if the target's quantity is above 1, we've removed the amount we expected, otherwise however many were left
- attributesRemoved[attribute] = newQuantity ? quantityToRemove : currentQuantity;
}
- await targetActor.update(updates);
+ const itemsCreated = await targetActor.createEmbeddedDocuments("Item", itemsToCreate);
+ await targetActor.updateEmbeddedDocuments("Item", itemsToUpdate);
- await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ATTRIBUTE.REMOVE, targetUuid, attributesRemoved);
+ itemsCreated.forEach(item => itemsAdded.push(item.toObject()));
+
+ await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ITEM.ADD, targetUuid, itemsAdded);
if (!isTransfer) {
const macroData = {
- action: "removeAttributes",
+ action: "addItems",
target: targetUuid,
- attributes: attributesRemoved
+ items: itemsAdded
};
+
await API._executeItemPileMacro(targetUuid, macroData);
- const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(targetUuid);
- await API.rerenderItemPileInventoryApplication(targetUuid, shouldBeDeleted);
+ await API.rerenderItemPileInventoryApplication(targetUuid);
- if (shouldBeDeleted) {
- await API._deleteItemPile(targetUuid);
- }
}
- return attributesRemoved;
+ return itemsAdded;
}
/**
- * Adds to attributes on an actor
- *
- * @param {Actor/Token/TokenDocument} target The target whose attribute will have a set quantity added to it
- * @param {object} attributes An object with each key being an attribute path, and its value being the quantity to add
+ * Subtracts the quantity of items on an actor. If the quantity of an item reaches 0, the item is removed from the actor.
*
- * @returns {Promise} Returns an array containing a key value pair of the attribute path and the quantity of that attribute that was removed
+ * @param {Actor/Token/TokenDocument} target The target to remove a items from
+ * @param {array} items An array of objects each containing the item id (key "_id") and the quantity to remove (key "quantity"), or an array of IDs to remove all quantities of
+ * @param {array/boolean} [itemTypeFilters=false] Array of item types disallowed - will default to module settings if none provided
*
+ * @returns {Promise} An array containing the objects of each item that was removed, with their quantities set to the number removed
*/
- static async addAttributes(target, attributes) {
+ static async removeItems(target, items, { itemTypeFilters = false } = {}) {
- const hookResult = Hooks.call(HOOKS.ATTRIBUTE.PRE_ADD, target, attributes);
+ const hookResult = Hooks.call(HOOKS.ITEM.PRE_REMOVE, target, items);
if (hookResult === false) return;
const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`AddAttributes | Could not determine the UUID, please provide a valid target`, true)
+ if (!targetUuid) throw lib.custom_error(`RemoveItems | Could not determine the UUID, please provide a valid target`, true);
- const targetActor = target instanceof TokenDocument
- ? target.actor
- : target;
+ if (itemTypeFilters) {
+ itemTypeFilters.forEach(filter => {
+ if (typeof filter !== "string") throw lib.custom_error(`RemoveItems | entries in the itemTypeFilters must be of type string`);
+ })
+ }
- Object.entries(attributes).forEach(entry => {
- const [attribute, quantity] = entry;
- if (!hasProperty(targetActor.data, attribute)) {
- throw lib.custom_error(`AddAttributes | Could not find attribute ${attribute} on target's actor with UUID "${targetUuid}"`, true)
+ const targetActorItems = API.getItemPileItems(target);
+
+ items.forEach(item => {
+ const itemId = typeof item === "string" ? item : item._id;
+ const actorItem = targetActorItems.find(actorItem => actorItem.id === itemId);
+ if (!actorItem) {
+ throw lib.custom_error(`RemoveItems | Could not find item with id "${itemId}" on target "${targetUuid}"`, true)
}
- if (!lib.is_real_number(quantity) && quantity > 0) {
- throw lib.custom_error(`AddAttributes | Attribute "${attribute}" must be of type number and greater than 0`, true)
+ const disallowedType = API.isItemTypeDisallowed(actorItem, itemTypeFilters);
+ if (disallowedType) {
+ throw lib.custom_error(`RemoveItems | Could not transfer item of type "${disallowedType}"`, true)
}
});
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.ADD_ATTRIBUTE, targetUuid, attributes);
-
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.REMOVE_ITEMS, targetUuid, items);
}
/**
* @private
*/
- static async _addAttributes(targetUuid, attributes, { isTransfer = false } = {}) {
+ static async _removeItems(targetUuid, items, { isTransfer = false } = {}) {
const target = await fromUuid(targetUuid);
const targetActor = target instanceof TokenDocument
? target.actor
: target;
- const updates = {};
- const attributesAdded = {};
+ const itemsRemoved = [];
+ const itemsToUpdate = [];
+ const itemsToDelete = [];
+ for (const item of items) {
- for (const [attribute, quantityToAdd] of Object.entries(attributes)) {
+ const itemId = typeof item === "string" ? item : item._id;
- const currentQuantity = getProperty(targetActor.data, attribute);
+ const actorItem = targetActor.items.get(itemId);
+ const removedItem = actorItem.toObject();
- updates[attribute] = currentQuantity + quantityToAdd;
- attributesAdded[attribute] = currentQuantity + quantityToAdd;
+ const currentQuantity = getProperty(actorItem.data, API.ITEM_QUANTITY_ATTRIBUTE);
+
+ const quantityToRemove = getProperty(item, API.ITEM_QUANTITY_ATTRIBUTE) ?? item.quantity ?? currentQuantity;
+
+ const newQuantity = Math.max(0, currentQuantity - quantityToRemove);
+
+ if (newQuantity >= 1) {
+ setProperty(removedItem, API.ITEM_QUANTITY_ATTRIBUTE, quantityToRemove);
+ itemsToUpdate.push({ _id: actorItem.id, [API.ITEM_QUANTITY_ATTRIBUTE]: newQuantity });
+ } else {
+ setProperty(removedItem, API.ITEM_QUANTITY_ATTRIBUTE, currentQuantity);
+ itemsToDelete.push(actorItem.id);
+ }
+
+ itemsRemoved.push(removedItem);
}
- await targetActor.update(updates);
+ await targetActor.updateEmbeddedDocuments("Item", itemsToUpdate);
+ await targetActor.deleteEmbeddedDocuments("Item", itemsToDelete);
- await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ATTRIBUTE.ADD, targetUuid, attributesAdded);
+ await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ITEM.REMOVE, targetUuid, itemsRemoved);
- if (isTransfer) {
+ if (!isTransfer) {
const macroData = {
- action: "addAttributes",
+ action: "removeItems",
target: targetUuid,
- attributes: attributesAdded
+ items: itemsRemoved
};
+
await API._executeItemPileMacro(targetUuid, macroData);
- await API.rerenderItemPileInventoryApplication(targetUuid);
+ const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(targetUuid);
+
+ await API.rerenderItemPileInventoryApplication(targetUuid, shouldBeDeleted);
+
+ if (shouldBeDeleted) {
+ await API._deleteItemPile(targetUuid);
+ }
+
}
- return attributesAdded;
+ return itemsRemoved;
}
-
/**
- * Transfers all dynamic attributes from a source to a target, removing it or subtracting from the source and adding them to the target
+ * Transfers items from the source to the target, subtracting a number of quantity from the source's item and adding it to the target's item, deleting items from the source if their quantity reaches 0
*
- * @param {Actor/Token/TokenDocument} source The source to transfer the attributes from
- * @param {Actor/Token/TokenDocument} target The target to transfer the attributes to
+ * @param {Actor/Token/TokenDocument} source The source to transfer the items from
+ * @param {Actor/Token/TokenDocument} target The target to transfer the items to
+ * @param {array} items An array of objects each containing the item id (key "_id") and the quantity to transfer (key "quantity")
+ * @param {array/boolean} [itemTypeFilters=false] Array of item types disallowed - will default to module settings if none provided
*
- * @returns {Promise} An object containing a key value pair of each attribute transferred, the key being the attribute path and its value being the quantity that was transferred
+ * @returns {Promise} An object containing a key value pair for each item added to the target, key being item ID, value being quantities added
*/
- static async transferAllAttributes(source, target) {
+ static async transferItems(source, target, items, { itemTypeFilters = false } = {}) {
- const hookResult = Hooks.call(HOOKS.ATTRIBUTE.PRE_TRANSFER_ALL, source, target);
+ const hookResult = Hooks.call(HOOKS.ITEM.PRE_TRANSFER, source, target, items);
if (hookResult === false) return;
const sourceUuid = lib.getUuid(source);
- if (!sourceUuid) throw lib.custom_error(`TransferAllAttributes | Could not determine the UUID, please provide a valid source`, true);
+ if (!sourceUuid) throw lib.custom_error(`TransferItems | Could not determine the UUID, please provide a valid source`, true)
+
+ if (itemTypeFilters) {
+ itemTypeFilters.forEach(filter => {
+ if (typeof filter !== "string") throw lib.custom_error(`TransferItems | entries in the itemTypeFilters must be of type string`);
+ })
+ }
+
+ const sourceActorItems = API.getItemPileItems(source);
+
+ items.forEach(item => {
+ const actorItem = sourceActorItems.find(actorItem => actorItem.id === item._id);
+ if (!actorItem) {
+ throw lib.custom_error(`TransferItems | Could not find item with id "${item._id}" on source "${sourceUuid}"`, true)
+ }
+ const disallowedType = API.isItemTypeDisallowed(actorItem, itemTypeFilters);
+ if (disallowedType) {
+ throw lib.custom_error(`TransferItems | Could not transfer item of type "${disallowedType}"`, true)
+ }
+ });
const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`TransferAllAttributes | Could not determine the UUID, please provide a valid target`, true);
+ if (!targetUuid) throw lib.custom_error(`TransferItems | Could not determine the UUID, please provide a valid target`, true)
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.TRANSFER_ALL_ATTRIBUTES, sourceUuid, targetUuid);
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.TRANSFER_ITEMS, sourceUuid, targetUuid, items, { itemTypeFilters });
}
/**
* @private
*/
- static async _transferAllAttributes(sourceUuid, targetUuid, { isEverything = false } = {}) {
-
- const source = await fromUuid(sourceUuid);
-
- const sourceActor = source instanceof TokenDocument
- ? source.actor
- : source;
-
- const target = await fromUuid(targetUuid);
-
- const targetActor = target instanceof TokenDocument
- ? target.actor
- : target;
-
- const sourceAttributes = API.getItemPileAttributes(sourceActor);
-
- const attributesToTransfer = sourceAttributes.filter(attribute => {
- return hasProperty(sourceActor.data, attribute.path)
- && getProperty(sourceActor.data, attribute.path) > 0
- && hasProperty(targetActor.data, attribute.path);
- }).map(attribute => attribute.path);
+ static async _transferItems(sourceUuid, targetUuid, items, { itemTypeFilters = false, isEverything = false } = {}) {
- const attributesRemoved = await API._removeAttributes(sourceUuid, attributesToTransfer, { isTransfer: true });
- const attributesAdded = await API._addAttributes(targetUuid, attributesRemoved, { isTransfer: true });
+ const itemsRemoved = await API._removeItems(sourceUuid, items, { itemTypeFilters, isTransfer: true });
- await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ATTRIBUTE.TRANSFER_ALL, sourceUuid, targetUuid, attributesAdded);
+ const itemsAdded = await API._addItems(targetUuid, itemsRemoved, { itemTypeFilters, isTransfer: true });
if (!isEverything) {
+ await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ITEM.TRANSFER, sourceUuid, targetUuid, itemsAdded);
+
const macroData = {
- action: "transferAllAttributes",
+ action: "transferItems",
source: sourceUuid,
target: targetUuid,
- attributes: attributesAdded
+ itemsAdded: itemsAdded
};
await API._executeItemPileMacro(sourceUuid, macroData);
await API._executeItemPileMacro(targetUuid, macroData);
const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(sourceUuid);
await API.rerenderItemPileInventoryApplication(sourceUuid, shouldBeDeleted);
+ await API.rerenderItemPileInventoryApplication(targetUuid);
if (shouldBeDeleted) {
await API._deleteItemPile(sourceUuid);
@@ -978,651 +1080,549 @@ export default class API {
}
- return attributesAdded;
+ return itemsAdded;
}
/**
- * Transfers all items and attributes between the source and the target.
+ * Transfers all items between the source and the target.
*
- * @param {Actor/Token/TokenDocument} source The actor to transfer all items and attributes from
- * @param {Actor/Token/TokenDocument} target The actor to receive all the items and attributes
+ * @param {Actor/Token/TokenDocument} source The actor to transfer all items from
+ * @param {Actor/Token/TokenDocument} target The actor to receive all the items
* @param {array/boolean} [itemTypeFilters=false] Array of item types disallowed - will default to module settings if none provided
*
- * @returns {Promise} An object containing all items and attributes transferred to the target
+ * @returns {Promise} An array containing all of the items that were transferred to the target
*/
- static async transferEverything(source, target, { itemTypeFilters = false } = {}) {
+ static async transferAllItems(source, target, { itemTypeFilters = false } = {}) {
- const hookResult = Hooks.call(HOOKS.PRE_TRANSFER_EVERYTHING, source, target, itemTypeFilters);
+ const hookResult = Hooks.call(HOOKS.ITEM.PRE_TRANSFER_ALL, source, target, itemTypeFilters);
if (hookResult === false) return;
const sourceUuid = lib.getUuid(source);
- if (!sourceUuid) throw lib.custom_error(`TransferEverything | Could not determine the UUID, please provide a valid source`, true)
+ if (!sourceUuid) throw lib.custom_error(`TransferAllItems | Could not determine the UUID, please provide a valid source`, true)
const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`TransferEverything | Could not determine the UUID, please provide a valid target`, true)
+ if (!targetUuid) throw lib.custom_error(`TransferAllItems | Could not determine the UUID, please provide a valid target`, true)
if (itemTypeFilters) {
itemTypeFilters.forEach(filter => {
- if (typeof filter !== "string") throw lib.custom_error(`TransferEverything | entries in the itemTypeFilters must be of type string`);
+ if (typeof filter !== "string") throw lib.custom_error(`RevertFromItemPile | entries in the itemTypeFilters must be of type string`);
})
}
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.TRANSFER_EVERYTHING, sourceUuid, targetUuid, { itemTypeFilters })
-
- }
-
- /**
- * @private
- */
- static async _transferEverything(sourceUuid, targetUuid, { itemTypeFilters = false } = {}) {
-
- const itemsTransferred = await API._transferAllItems(sourceUuid, targetUuid, {
- itemTypeFilters,
- isEverything: true
- });
- const attributesTransferred = await API._transferAllAttributes(sourceUuid, targetUuid, { isEverything: true });
-
- await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.TRANSFER_EVERYTHING, sourceUuid, targetUuid, itemsTransferred, attributesTransferred);
-
- const macroData = {
- action: "transferEverything",
- source: sourceUuid,
- target: targetUuid,
- items: itemsTransferred,
- attributes: attributesTransferred
- };
- await API._executeItemPileMacro(sourceUuid, macroData);
- await API._executeItemPileMacro(targetUuid, macroData);
-
- const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(sourceUuid);
- await API.rerenderItemPileInventoryApplication(sourceUuid, shouldBeDeleted);
- await API.rerenderItemPileInventoryApplication(targetUuid);
-
- if (shouldBeDeleted) {
- await API._deleteItemPile(sourceUuid);
- }
-
- return {
- itemsTransferred,
- attributesTransferred
- };
-
- }
-
- /**
- * Turns a token and its actor into an item pile
- *
- * @param {Token/TokenDocument} target The target to be turned into an item pile
- * @param {object} pileSettings Overriding settings to be put on the item pile's settings
- * @param {object} tokenSettings Overriding settings that will update the token
- *
- * @return {Promise} The uuid of the target after it was turned into an item pile
- */
- static async turnTokenIntoItemPile(target, { pileSettings = {}, tokenSettings = {} } = {}) {
- const hookResult = Hooks.call(HOOKS.PILE.PRE_TURN_INTO, target, pileSettings, tokenSettings);
- if (hookResult === false) return;
- const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`TurnIntoItemPile | Could not determine the UUID, please provide a valid target`, true)
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.TURN_INTO_PILE, targetUuid, pileSettings, tokenSettings);
- }
-
- /**
- * @private
- */
- static async _turnTokenIntoItemPile(targetUuid, pileSettings = {}, tokenSettings = {}) {
-
- const target = await fromUuid(targetUuid);
-
- const existingPileSettings = foundry.utils.mergeObject(CONSTANTS.PILE_DEFAULTS, lib.getItemPileData(target));
- const newPileSettings = foundry.utils.mergeObject(existingPileSettings, pileSettings);
- newPileSettings.enabled = true;
-
- await API._updateItemPile(targetUuid, newPileSettings, { tokenSettings });
-
- setTimeout(API.rerenderTokenHud, 100);
-
- await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.PILE.TURN_INTO, targetUuid, newPileSettings);
-
- return targetUuid;
-
- }
-
- /**
- * Reverts a token from an item pile into a normal token and actor
- *
- * @param {Token/TokenDocument} target The target to be reverted from an item pile
- * @param {object} tokenSettings Overriding settings that will update the token
- *
- * @return {Promise} The uuid of the target after it was reverted from an item pile
- */
- static async revertTokenFromItemPile(target, { tokenSettings = {} } = {}) {
- const hookResult = Hooks.call(HOOKS.PILE.PRE_REVERT_FROM, target, tokenSettings);
- if (hookResult === false) return;
- const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`RevertFromItemPile | Could not determine the UUID, please provide a valid target`, true)
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.REVERT_FROM_PILE, targetUuid, tokenSettings);
+ return itemPileSocket.executeAsGM(
+ SOCKET_HANDLERS.TRANSFER_ALL_ITEMS,
+ sourceUuid,
+ targetUuid,
+ {
+ itemTypeFilters
+ }
+ );
}
/**
* @private
*/
- static async _revertTokenFromItemPile(targetUuid, tokenSettings) {
+ static async _transferAllItems(sourceUuid, targetUuid, { itemTypeFilters = false, isEverything = false } = {}) {
+
+ const source = await fromUuid(sourceUuid);
+ if (!source) throw lib.custom_error(`TransferAllItems | Could not find source with UUID "${sourceUuid}"`, true)
const target = await fromUuid(targetUuid);
+ if (!target) throw lib.custom_error(`TransferAllItems | Could not find target with UUID "${targetUuid}"`, true)
- const pileSettings = foundry.utils.mergeObject(CONSTANTS.PILE_DEFAULTS, lib.getItemPileData(target));
+ if (!Array.isArray(itemTypeFilters)) {
+ itemTypeFilters = API.ITEM_TYPE_FILTERS;
+ } else {
+ itemTypeFilters.forEach(filter => {
+ if (typeof filter !== "string") throw lib.custom_error(`TransferAllItems | entries in the itemTypeFilters must be of type string`);
+ })
+ }
- pileSettings.enabled = false;
+ const itemsToRemove = API.getItemPileItems(target, itemTypeFilters).map(item => item.toObject());
- await API._updateItemPile(targetUuid, pileSettings, { tokenSettings });
+ const itemsRemoved = await API._removeItems(sourceUuid, itemsToRemove, { itemTypeFilters, isTransfer: true });
+ const itemsAdded = await API._addItems(targetUuid, itemsRemoved, { itemTypeFilters, isTransfer: true });
- setTimeout(API.rerenderTokenHud, 100);
+ if (!isEverything) {
- if (target instanceof TokenDocument) {
- await API.rerenderItemPileInventoryApplication(targetUuid);
- }
+ await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.TRANSFER_ALL_ITEMS, HOOKS.ITEM.TRANSFER_ALL, sourceUuid, targetUuid, itemsAdded);
- await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.PILE.REVERT_FROM, targetUuid);
+ const macroData = {
+ action: "transferAllItems",
+ source: sourceUuid,
+ target: targetUuid,
+ items: itemsAdded
+ };
+ await API._executeItemPileMacro(sourceUuid, macroData);
+ await API._executeItemPileMacro(targetUuid, macroData);
- return targetUuid;
+ const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(sourceUuid);
+ await API.rerenderItemPileInventoryApplication(sourceUuid, shouldBeDeleted);
+ await API.rerenderItemPileInventoryApplication(targetUuid);
- }
+ if (shouldBeDeleted) {
+ await API._deleteItemPile(sourceUuid);
+ }
- /**
- * Causes every user's token HUD to rerender
- *
- * @return {Promise}
- */
- static async rerenderTokenHud() {
- return itemPileSocket.executeForEveryone(SOCKET_HANDLERS.RERENDER_TOKEN_HUD);
- }
+ }
- /**
- * @private
- */
- static async _rerenderTokenHud() {
- if (!canvas.tokens.hud.rendered) return;
- await canvas.tokens.hud.render(true)
- return true;
+ return itemsAdded;
}
/**
- * Opens a pile if it is enabled and a container
- *
- * @param {Token/TokenDocument} target
- * @param {Token/TokenDocument/boolean} [interactingToken=false]
+ * Adds to attributes on an actor
*
- * @return {Promise}
- */
- static async openItemPile(target, interactingToken = false) {
- const data = lib.getItemPileData(target);
- if (!data?.enabled || !data?.isContainer) return false;
- const wasLocked = data.locked;
- data.closed = false;
- data.locked = false;
- if (wasLocked) {
- const hookResult = Hooks.call(HOOKS.PILE.PRE_UNLOCK, target, data, interactingToken);
- if (hookResult === false) return;
- }
- const hookResult = Hooks.call(HOOKS.PILE.PRE_OPEN, target, data, interactingToken);
- if (hookResult === false) return;
- if (data.openSound) {
- AudioHelper.play({ src: data.openSound })
- }
- return API.updateItemPile(target, data, { interactingToken });
- }
-
- /**
- * Closes a pile if it is enabled and a container
+ * @param {Actor/Token/TokenDocument} target The target whose attribute will have a set quantity added to it
+ * @param {object} attributes An object with each key being an attribute path, and its value being the quantity to add
*
- * @param {Token/TokenDocument} target Target pile to close
- * @param {Token/TokenDocument/boolean} [interactingToken=false]
+ * @returns {Promise} Returns an array containing a key value pair of the attribute path and the quantity of that attribute that was removed
*
- * @return {Promise}
*/
- static async closeItemPile(target, interactingToken = false) {
- const data = lib.getItemPileData(target);
- if (!data?.enabled || !data?.isContainer) return false;
- data.closed = true;
- const hookResult = Hooks.call(HOOKS.PILE.PRE_CLOSE, target, data, interactingToken);
+ static async addAttributes(target, attributes) {
+
+ const hookResult = Hooks.call(HOOKS.ATTRIBUTE.PRE_ADD, target, attributes);
if (hookResult === false) return;
- if (data.closeSound) {
- AudioHelper.play({ src: data.closeSound })
- }
- return API.updateItemPile(target, data, interactingToken);
- }
- /**
- * Toggles a pile's closed state if it is enabled and a container
- *
- * @param {Token/TokenDocument} target Target pile to open or close
- * @param {Token/TokenDocument/boolean} [interactingToken=false]
- *
- * @return {Promise}
- */
- static async toggleItemPileClosed(target, interactingToken = false) {
- const data = lib.getItemPileData(target);
- if (!data?.enabled || !data?.isContainer) return false;
- if (data.closed) {
- await API.openItemPile(target, interactingToken);
- } else {
- await API.closeItemPile(target, interactingToken);
- }
- return !data.closed;
- }
+ const targetUuid = lib.getUuid(target);
+ if (!targetUuid) throw lib.custom_error(`AddAttributes | Could not determine the UUID, please provide a valid target`, true)
- /**
- * Locks a pile if it is enabled and a container
- *
- * @param {Token/TokenDocument} target Target pile to lock
- * @param {Token/TokenDocument/boolean} [interactingToken=false]
- *
- * @return {Promise}
- */
- static async lockItemPile(target, interactingToken = false) {
- const data = lib.getItemPileData(target);
- if (!data?.enabled || !data?.isContainer) return false;
- const wasClosed = data.closed;
- data.closed = true;
- data.locked = true;
- if (!wasClosed) {
- const hookResult = Hooks.call(HOOKS.PILE.PRE_CLOSE, target, data, interactingToken);
- if (hookResult === false) return;
- }
- const hookResult = Hooks.call(HOOKS.PILE.PRE_LOCK, target, data, interactingToken);
- if (hookResult === false) return;
- if (data.closeSound && !wasClosed) {
- AudioHelper.play({ src: data.closeSound })
- }
- return API.updateItemPile(target, data, interactingToken);
- }
+ const targetActor = target instanceof TokenDocument
+ ? target.actor
+ : target;
- /**
- * Unlocks a pile if it is enabled and a container
- *
- * @param {Token/TokenDocument} target Target pile to unlock
- * @param {Token/TokenDocument/boolean} [interactingToken=false]
- *
- * @return {Promise}
- */
- static async unlockItemPile(target, interactingToken = false) {
- const data = lib.getItemPileData(target);
- if (!data?.enabled || !data?.isContainer) return false;
- data.locked = false;
- Hooks.call(HOOKS.PILE.PRE_UNLOCK, target, data, interactingToken);
- return API.updateItemPile(target, data, interactingToken);
- }
+ Object.entries(attributes).forEach(entry => {
+ const [attribute, quantity] = entry;
+ if (!hasProperty(targetActor.data, attribute)) {
+ throw lib.custom_error(`AddAttributes | Could not find attribute ${attribute} on target's actor with UUID "${targetUuid}"`, true)
+ }
+ if (!lib.is_real_number(quantity) && quantity > 0) {
+ throw lib.custom_error(`AddAttributes | Attribute "${attribute}" must be of type number and greater than 0`, true)
+ }
+ });
+
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.ADD_ATTRIBUTE, targetUuid, attributes);
- /**
- * Toggles a pile's locked state if it is enabled and a container
- *
- * @param {Token/TokenDocument} target Target pile to lock or unlock
- * @param {Token/TokenDocument/boolean} [interactingToken=false]
- *
- * @return {Promise}
- */
- static async toggleItemPileLocked(target, interactingToken = false) {
- const data = lib.getItemPileData(target);
- if (!data?.enabled || !data?.isContainer) return false;
- if (data.locked) {
- return API.unlockItemPile(target, interactingToken);
- }
- return API.lockItemPile(target, interactingToken);
}
/**
- * Causes the item pile to play a sound as it was attempted to be opened, but was locked
- *
- * @param {Token/TokenDocument} target
- *
- * @return {Promise}
+ * @private
*/
- static async rattleItemPile(target) {
- const data = lib.getItemPileData(target);
- if (!data?.enabled || !data?.isContainer) return false;
- if (data.locked && data.lockedSound) {
- AudioHelper.play({ src: data.lockedSound })
+ static async _addAttributes(targetUuid, attributes, { isTransfer = false } = {}) {
+
+ const target = await fromUuid(targetUuid);
+ const targetActor = target instanceof TokenDocument
+ ? target.actor
+ : target;
+
+ const updates = {};
+ const attributesAdded = {};
+
+ for (const [attribute, quantityToAdd] of Object.entries(attributes)) {
+
+ const currentQuantity = getProperty(targetActor.data, attribute);
+
+ updates[attribute] = currentQuantity + quantityToAdd;
+ attributesAdded[attribute] = currentQuantity + quantityToAdd;
+
}
- return true;
- }
- /**
- * Whether an item pile is locked. If it is not enabled or not a container, it is always false.
- *
- * @param {Token/TokenDocument} target
- *
- * @return {boolean}
- */
- static isItemPileLocked(target) {
- const data = lib.getItemPileData(target);
- if (!data?.enabled || !data?.isContainer) return false;
- return data.locked;
- }
+ await targetActor.update(updates);
+
+ await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ATTRIBUTE.ADD, targetUuid, attributesAdded);
+
+ if (isTransfer) {
+
+ const macroData = {
+ action: "addAttributes",
+ target: targetUuid,
+ attributes: attributesAdded
+ };
+ await API._executeItemPileMacro(targetUuid, macroData);
+
+ await API.rerenderItemPileInventoryApplication(targetUuid);
+ }
- /**
- * Whether an item pile is closed. If it is not enabled or not a container, it is always false.
- *
- * @param {Token/TokenDocument} target
- *
- * @return {boolean}
- */
- static isItemPileClosed(target) {
- const data = lib.getItemPileData(target);
- if (!data?.enabled || !data?.isContainer) return false;
- return data.closed;
- }
+ return attributesAdded;
- /**
- * Whether an item pile is a container. If it is not enabled, it is always false.
- *
- * @param {Token/TokenDocument} target
- *
- * @return {boolean}
- */
- static isItemPileContainer(target) {
- const data = lib.getItemPileData(target);
- return data?.enabled && data?.isContainer;
}
/**
- * Updates a pile with new data.
+ * Subtracts attributes on the target
*
- * @param {Token/TokenDocument} target
- * @param {object} newData
- * @param {Token/TokenDocument/boolean} [interactingToken=false]
- * @param {object/boolean} [tokenSettings=false]
+ * @param {Token/TokenDocument} target The target whose attributes will be subtracted from
+ * @param {array/object} attributes This can be either an array of attributes to subtract (to zero out a given attribute), or an object with each key being an attribute path, and its value being the quantity to subtract
*
- * @return {Promise}
+ * @returns {Promise} Returns an array containing a key value pair of the attribute path and the quantity of that attribute that was removed
*/
- static async updateItemPile(target, newData, { interactingToken = false, tokenSettings = false } = {}) {
+ static async removeAttributes(target, attributes) {
+
+ const hookResult = Hooks.call(HOOKS.ATTRIBUTE.PRE_REMOVE, target, attributes);
+ if (hookResult === false) return;
const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`updateItemPile | Could not determine the UUID, please provide a valid target`, true);
+ if (!targetUuid) throw lib.custom_error(`RemoveAttributes | Could not determine the UUID, please provide a valid target`, true)
- const interactingTokenUuid = interactingToken ? lib.getUuid(interactingToken) : false;
- if (interactingToken && !interactingTokenUuid) throw lib.custom_error(`updateItemPile | Could not determine the UUID, please provide a valid target`, true);
+ const targetActor = target instanceof TokenDocument
+ ? target.actor
+ : target;
- const hookResult = Hooks.call(HOOKS.PILE.PRE_UPDATE, target, newData, interactingToken, tokenSettings);
- if (hookResult === false) return;
+ if (Array.isArray(attributes)) {
+ attributes.forEach(attribute => {
+ if (typeof attribute !== "string") {
+ throw lib.custom_error(`RemoveAttributes | Each attribute in the array must be of type string`, true)
+ }
+ if (!hasProperty(targetActor.data, attribute)) {
+ throw lib.custom_error(`RemoveAttributes | Could not find attribute ${attribute} on target's actor with UUID "${targetUuid}"`, true)
+ }
+ });
+ } else {
+ Object.entries(attributes).forEach(entry => {
+ const [attribute, quantity] = entry;
+ if (!hasProperty(targetActor.data, attribute)) {
+ throw lib.custom_error(`RemoveAttributes | Could not find attribute ${attribute} on target's actor with UUID "${targetUuid}"`, true)
+ }
+ if (!lib.is_real_number(quantity) && quantity > 0) {
+ throw lib.custom_error(`RemoveAttributes | Attribute "${attribute}" must be of type number and greater than 0`, true)
+ }
+ });
+ }
+
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.REMOVE_ATTRIBUTES, targetUuid, attributes);
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.UPDATE_PILE, targetUuid, newData, {
- interactingTokenUuid,
- tokenSettings
- })
}
/**
* @private
*/
- static async _updateItemPile(targetUuid, newData, { interactingTokenUuid = false, tokenSettings = false } = {}) {
+ static async _removeAttributes(targetUuid, attributes, { isTransfer = false } = {}) {
const target = await fromUuid(targetUuid);
+ const targetActor = target instanceof TokenDocument
+ ? target.actor
+ : target;
- const oldData = lib.getItemPileData(target);
+ const updates = {};
+ const attributesRemoved = {};
- const data = foundry.utils.mergeObject(
- foundry.utils.duplicate(oldData),
- foundry.utils.duplicate(newData)
- );
+ if (Array.isArray(attributes)) {
+ attributes = Object.fromEntries(attributes.map(attribute => {
+ return [attribute, getProperty(targetActor.data, attribute)];
+ }))
+ }
- const diff = foundry.utils.diffObject(oldData, data);
+ for (const [attribute, quantityToRemove] of Object.entries(attributes)) {
- await lib.wait(25);
+ const currentQuantity = getProperty(targetActor.data, attribute);
+ const newQuantity = Math.max(0, currentQuantity - quantityToRemove);
- await lib.updateItemPile(target, data, tokenSettings);
+ updates[attribute] = newQuantity;
- if (data.isEnabled && data.isContainer) {
- if (diff?.closed === true) {
- await API._executeItemPileMacro(targetUuid, {
- action: "closeItemPile",
- source: interactingTokenUuid,
- target: targetUuid
- });
- }
- if (diff?.locked === true) {
- await API._executeItemPileMacro(targetUuid, {
- action: "lockItemPile",
- source: interactingTokenUuid,
- target: targetUuid
- });
- }
- if (diff?.locked === false) {
- await API._executeItemPileMacro(targetUuid, {
- action: "unlockItemPile",
- source: interactingTokenUuid,
- target: targetUuid
- });
- }
- if (diff?.closed === false) {
- await API._executeItemPileMacro(targetUuid, {
- action: "openItemPile",
- source: interactingTokenUuid,
- target: targetUuid
- });
- }
+ // if the target's quantity is above 1, we've removed the amount we expected, otherwise however many were left
+ attributesRemoved[attribute] = newQuantity ? quantityToRemove : currentQuantity;
}
- return itemPileSocket.executeForEveryone(SOCKET_HANDLERS.UPDATED_PILE, targetUuid, diff, interactingTokenUuid);
- }
-
- /**
- * @private
- */
- static async _updatedItemPile(targetUuid, diffData, interactingTokenUuid) {
-
- const target = await lib.getToken(targetUuid);
+ await targetActor.update(updates);
- const interactingToken = interactingTokenUuid ? await fromUuid(interactingTokenUuid) : false;
+ await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ATTRIBUTE.REMOVE, targetUuid, attributesRemoved);
- if (foundry.utils.isObjectEmpty(diffData)) return;
+ if (!isTransfer) {
- const data = lib.getItemPileData(target);
+ const macroData = {
+ action: "removeAttributes",
+ target: targetUuid,
+ attributes: attributesRemoved
+ };
+ await API._executeItemPileMacro(targetUuid, macroData);
- Hooks.callAll(HOOKS.PILE.UPDATE, target, diffData, interactingToken)
+ const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(targetUuid);
+ await API.rerenderItemPileInventoryApplication(targetUuid, shouldBeDeleted);
- if (data.isEnabled && data.isContainer) {
- if (diffData?.closed === true) {
- Hooks.callAll(HOOKS.PILE.CLOSE, target, interactingToken)
- }
- if (diffData?.locked === true) {
- Hooks.callAll(HOOKS.PILE.LOCK, target, interactingToken)
- }
- if (diffData?.locked === false) {
- Hooks.callAll(HOOKS.PILE.UNLOCK, target, interactingToken)
- }
- if (diffData?.closed === false) {
- Hooks.callAll(HOOKS.PILE.OPEN, target, interactingToken)
+ if (shouldBeDeleted) {
+ await API._deleteItemPile(targetUuid);
}
}
+
+ return attributesRemoved;
+
}
/**
- * Deletes a pile, calling the relevant hooks.
+ * Transfers a set quantity of an attribute from a source to a target, removing it or subtracting from the source and adds it the target
*
- * @param {Token/TokenDocument} target
+ * @param {Actor/Token/TokenDocument} source The source to transfer the attribute from
+ * @param {Actor/Token/TokenDocument} target The target to transfer the attribute to
+ * @param {array/object} attributes This can be either an array of attributes to transfer (to transfer all of a given attribute), or an object with each key being an attribute path, and its value being the quantity to transfer
*
- * @return {Promise}
+ * @returns {Promise} An object containing a key value pair of each attribute transferred, the key being the attribute path and its value being the quantity that was transferred
*/
- static async deleteItemPile(target) {
- if (!API.isValidItemPile(target)) {
- if (!targetUuid) throw lib.custom_error(`deleteItemPile | This is not an item pile, please provide a valid target`, true);
- }
+ static async transferAttributes(source, target, attributes) {
+
+ const hookResult = Hooks.call(HOOKS.ATTRIBUTE.PRE_TRANSFER, source, target, attributes);
+ if (hookResult === false) return;
+
+ const sourceUuid = lib.getUuid(source);
+ if (!sourceUuid) throw lib.custom_error(`TransferAttributes | Could not determine the UUID, please provide a valid source`, true)
+
const targetUuid = lib.getUuid(target);
- if (!targetUuid) throw lib.custom_error(`deleteItemPile | Could not determine the UUID, please provide a valid target`, true);
- if (!targetUuid.includes("Token")) {
- throw lib.custom_error(`deleteItemPile | Please provide a Token or TokenDocument`, true);
+ if (!targetUuid) throw lib.custom_error(`TransferAttributes | Could not determine the UUID, please provide a valid target`, true)
+
+ const sourceActor = source instanceof TokenDocument
+ ? source.actor
+ : source;
+
+ const targetActor = target instanceof TokenDocument
+ ? target.actor
+ : target;
+
+ if (Array.isArray(attributes)) {
+ attributes.forEach(attribute => {
+ if (typeof attribute !== "string") {
+ throw lib.custom_error(`TransferAttributes | Each attribute in the array must be of type string`, true)
+ }
+ if (!hasProperty(sourceActor.data, attribute)) {
+ throw lib.custom_error(`TransferAttributes | Could not find attribute ${attribute} on source's actor with UUID "${targetUuid}"`, true)
+ }
+ if (!hasProperty(targetActor.data, attribute)) {
+ throw lib.custom_error(`TransferAttributes | Could not find attribute ${attribute} on target's actor with UUID "${targetUuid}"`, true)
+ }
+ });
+ } else {
+ Object.entries(attributes).forEach(entry => {
+ const [attribute, quantity] = entry;
+ if (!hasProperty(sourceActor.data, attribute)) {
+ throw lib.custom_error(`TransferAttributes | Could not find attribute ${attribute} on source's actor with UUID "${targetUuid}"`, true)
+ }
+ if (!hasProperty(targetActor.data, attribute)) {
+ throw lib.custom_error(`TransferAttributes | Could not find attribute ${attribute} on target's actor with UUID "${targetUuid}"`, true)
+ }
+ if (!lib.is_real_number(quantity) && quantity > 0) {
+ throw lib.custom_error(`TransferAttributes | Attribute "${attribute}" must be of type number and greater than 0`, true)
+ }
+ });
}
- const hookResult = Hooks.call(HOOKS.PILE.PRE_DELETE, target);
- if (hookResult === false) return;
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.DELETE_PILE, targetUuid);
+
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.TRANSFER_ATTRIBUTES, sourceUuid, targetUuid, attributes);
+
}
- static async _deleteItemPile(targetUuid) {
- const target = await lib.getToken(targetUuid);
- return target.delete();
- }
+ /**
+ * @private
+ */
+ static async _transferAttributes(sourceUuid, targetUuid, attributes, { isEverything = false } = {}) {
+
+ const attributesRemoved = await API._removeAttributes(sourceUuid, attributes, { isTransfer: true });
+
+ const attributesAdded = await API._addAttributes(targetUuid, attributesRemoved, { isTransfer: true });
+
+ if (!isEverything) {
+
+ await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ATTRIBUTE.TRANSFER, sourceUuid, targetUuid, attributesAdded);
+
+ const macroData = {
+ action: "transferAttributes",
+ source: sourceUuid,
+ target: targetUuid,
+ attributes: attributesAdded
+ };
+ await API._executeItemPileMacro(sourceUuid, macroData);
+ await API._executeItemPileMacro(targetUuid, macroData);
+
+ const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(sourceUuid);
+ await API.rerenderItemPileInventoryApplication(sourceUuid, shouldBeDeleted);
+ await API.rerenderItemPileInventoryApplication(targetUuid);
- /* -------- UTILITY METHODS -------- */
+ if (shouldBeDeleted) {
+ await API._deleteItemPile(sourceUuid);
+ }
- /**
- * Checks whether an item (or item data) is of a type that is not allowed. If an array whether that type is allowed
- * or not, returning the type if it is NOT allowed.
- *
- * @param {Item/Object} item
- * @param {array/boolean} [itemTypeFilters=false]
- * @return {boolean/string}
- */
- static isItemTypeDisallowed(item, itemTypeFilters = false) {
- if (!API.ITEM_TYPE_ATTRIBUTE) return false;
- if (!Array.isArray(itemTypeFilters)) itemTypeFilters = API.ITEM_TYPE_FILTERS;
- const itemType = getProperty(item, API.ITEM_TYPE_ATTRIBUTE);
- if (itemTypeFilters.includes(itemType)) {
- return itemType;
}
- return false;
- }
- /**
- * Whether a given document is a valid pile or not
- *
- * @param {TokenDocument|Actor} document
- * @return {boolean}
- */
- static isValidItemPile(document) {
- const documentActor = document instanceof TokenDocument ? document.actor : document;
- return document && !document.destroyed && documentActor && lib.getItemPileData(document)?.enabled;
+ return attributesAdded
+
}
/**
- * Whether the item pile is empty
+ * Transfers all dynamic attributes from a source to a target, removing it or subtracting from the source and adding them to the target
*
- * @param {TokenDocument|Actor} target
- * @returns {boolean}
+ * @param {Actor/Token/TokenDocument} source The source to transfer the attributes from
+ * @param {Actor/Token/TokenDocument} target The target to transfer the attributes to
+ *
+ * @returns {Promise} An object containing a key value pair of each attribute transferred, the key being the attribute path and its value being the quantity that was transferred
*/
- static isItemPileEmpty(target){
- const targetActor = target instanceof TokenDocument
- ? target.actor
- : target;
+ static async transferAllAttributes(source, target) {
- if(!targetActor) return false;
+ const hookResult = Hooks.call(HOOKS.ATTRIBUTE.PRE_TRANSFER_ALL, source, target);
+ if (hookResult === false) return;
- const hasNoItems = API.getItemPileItems(target).length === 0;
+ const sourceUuid = lib.getUuid(source);
+ if (!sourceUuid) throw lib.custom_error(`TransferAllAttributes | Could not determine the UUID, please provide a valid source`, true);
- const attributes = API.getItemPileAttributes(target);
- const hasEmptyAttributes = attributes.find(attribute => {
- return hasProperty(targetActor.data, attribute.path) && getProperty(targetActor.data, attribute.path) === 0;
- })
- return hasNoItems && hasEmptyAttributes;
- }
+ const targetUuid = lib.getUuid(target);
+ if (!targetUuid) throw lib.custom_error(`TransferAllAttributes | Could not determine the UUID, please provide a valid target`, true);
+
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.TRANSFER_ALL_ATTRIBUTES, sourceUuid, targetUuid);
- /**
- * Returns the item type filters for a given item pile
- *
- * @param target
- * @returns {Array}
- */
- static getItemPileItemTypeFilters(target){
- if(!API.isValidItemPile(target)) return [];
- const pileData = lib.getItemPileData(target);
- return pileData.itemTypeFilters
- ? pileData.itemTypeFilters.split(',').map(str => str.trim().toLowerCase())
- : API.ITEM_TYPE_FILTERS;
}
/**
- * Returns the items this item pile can transfer
- *
- * @param {TokenDocument|Actor} target
- * @param {array/boolean} [itemTypeFilters=false] Array of item types disallowed - will default to pile settings or module settings if none provided
- * @returns {Array}
+ * @private
*/
- static getItemPileItems(target, itemTypeFilters = false){
+ static async _transferAllAttributes(sourceUuid, targetUuid, { isEverything = false } = {}) {
- if(!API.isValidItemPile(target)) return [];
+ const source = await fromUuid(sourceUuid);
- const pileItemFilters = Array.isArray(itemTypeFilters)
- ? new Set(itemTypeFilters)
- : new Set(API.getItemPileItemTypeFilters(target));
+ const sourceActor = source instanceof TokenDocument
+ ? source.actor
+ : source;
+
+ const target = await fromUuid(targetUuid);
const targetActor = target instanceof TokenDocument
? target.actor
: target;
- return Array.from(targetActor.items).filter(item => {
- const itemType = getProperty(item.data, API.ITEM_TYPE_ATTRIBUTE);
- return !pileItemFilters.has(itemType);
- })
+ const sourceAttributes = API.getItemPileAttributes(sourceActor);
- }
+ const attributesToTransfer = sourceAttributes.filter(attribute => {
+ return hasProperty(sourceActor.data, attribute.path)
+ && getProperty(sourceActor.data, attribute.path) > 0
+ && hasProperty(targetActor.data, attribute.path);
+ }).map(attribute => attribute.path);
+
+ const attributesRemoved = await API._removeAttributes(sourceUuid, attributesToTransfer, { isTransfer: true });
+ const attributesAdded = await API._addAttributes(targetUuid, attributesRemoved, { isTransfer: true });
+
+ await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.ATTRIBUTE.TRANSFER_ALL, sourceUuid, targetUuid, attributesAdded);
+
+ if (!isEverything) {
+
+ const macroData = {
+ action: "transferAllAttributes",
+ source: sourceUuid,
+ target: targetUuid,
+ attributes: attributesAdded
+ };
+ await API._executeItemPileMacro(sourceUuid, macroData);
+ await API._executeItemPileMacro(targetUuid, macroData);
+
+ const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(sourceUuid);
+ await API.rerenderItemPileInventoryApplication(sourceUuid, shouldBeDeleted);
+
+ if (shouldBeDeleted) {
+ await API._deleteItemPile(sourceUuid);
+ }
+
+ }
+
+ return attributesAdded;
- /**
- * Returns the attributes this item pile can transfer
- *
- * @param {TokenDocument|Actor} target
- * @returns {array}
- */
- static getItemPileAttributes(target){
- const pileData = lib.getItemPileData(target);
- return pileData.overrideAttributes || API.DYNAMIC_ATTRIBUTES;
}
/**
- * Refreshes the target image of an item pile, ensuring it remains in sync
+ * Transfers all items and attributes between the source and the target.
*
- * @param target
- * @return {Promise}
+ * @param {Actor/Token/TokenDocument} source The actor to transfer all items and attributes from
+ * @param {Actor/Token/TokenDocument} target The actor to receive all the items and attributes
+ * @param {array/boolean} [itemTypeFilters=false] Array of item types disallowed - will default to module settings if none provided
+ *
+ * @returns {Promise} An object containing all items and attributes transferred to the target
*/
- static async refreshItemPile(target) {
- if (!API.isValidItemPile(target)) return;
+ static async transferEverything(source, target, { itemTypeFilters = false } = {}) {
+
+ const hookResult = Hooks.call(HOOKS.PRE_TRANSFER_EVERYTHING, source, target, itemTypeFilters);
+ if (hookResult === false) return;
+
+ const sourceUuid = lib.getUuid(source);
+ if (!sourceUuid) throw lib.custom_error(`TransferEverything | Could not determine the UUID, please provide a valid source`, true)
+
const targetUuid = lib.getUuid(target);
- return itemPileSocket.executeAsGM(SOCKET_HANDLERS.REFRESH_PILE, targetUuid)
+ if (!targetUuid) throw lib.custom_error(`TransferEverything | Could not determine the UUID, please provide a valid target`, true)
+
+ if (itemTypeFilters) {
+ itemTypeFilters.forEach(filter => {
+ if (typeof filter !== "string") throw lib.custom_error(`TransferEverything | entries in the itemTypeFilters must be of type string`);
+ })
+ }
+
+ return itemPileSocket.executeAsGM(SOCKET_HANDLERS.TRANSFER_EVERYTHING, sourceUuid, targetUuid, { itemTypeFilters })
+
}
/**
* @private
*/
- static async _refreshItemPile(targetUuid) {
- const targetDocument = await fromUuid(targetUuid);
+ static async _transferEverything(sourceUuid, targetUuid, { itemTypeFilters = false } = {}) {
- if (!API.isValidItemPile(targetDocument)) return;
+ const itemsTransferred = await API._transferAllItems(sourceUuid, targetUuid, {
+ itemTypeFilters,
+ isEverything: true
+ });
+ const attributesTransferred = await API._transferAllAttributes(sourceUuid, targetUuid, { isEverything: true });
- let targets = [targetDocument]
- if (targetDocument instanceof Actor) {
- targets = Array.from(canvas.tokens.getDocuments()).filter(token => token.actor === targetDocument);
+ await itemPileSocket.executeForEveryone(SOCKET_HANDLERS.CALL_HOOK, HOOKS.TRANSFER_EVERYTHING, sourceUuid, targetUuid, itemsTransferred, attributesTransferred);
+
+ const macroData = {
+ action: "transferEverything",
+ source: sourceUuid,
+ target: targetUuid,
+ items: itemsTransferred,
+ attributes: attributesTransferred
+ };
+ await API._executeItemPileMacro(sourceUuid, macroData);
+ await API._executeItemPileMacro(targetUuid, macroData);
+
+ const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(sourceUuid);
+ await API.rerenderItemPileInventoryApplication(sourceUuid, shouldBeDeleted);
+ await API.rerenderItemPileInventoryApplication(targetUuid);
+
+ if (shouldBeDeleted) {
+ await API._deleteItemPile(sourceUuid);
}
- return Promise.allSettled(targets.map(_target => {
- return new Promise(async (resolve) => {
- const uuid = lib.getUuid(_target);
- const shouldBeDeleted = await API._checkItemPileShouldBeDeleted(uuid);
- if (!shouldBeDeleted) {
- await _target.update({
- "img": API._getItemPileTokenImage(targetDocument),
- "scale": API._getItemPileTokenScale(targetDocument),
- })
- }
- resolve();
- })
- }));
+ return {
+ itemsTransferred,
+ attributesTransferred
+ };
+
}
+ /* -------- UTILITY METHODS -------- */
+
/**
- * Causes all connected users to re-render a specific pile's inventory UI
+ * Causes every user's token HUD to rerender
*
- * @param {string} inPileUuid The uuid of the pile to be re-rendered
- * @param {boolean} [deleted=false] Whether the pile was deleted as a part of this re-render
* @return {Promise}
*/
- static async rerenderItemPileInventoryApplication(inPileUuid, deleted = false) {
- return itemPileSocket.executeForEveryone(SOCKET_HANDLERS.RERENDER_PILE_INVENTORY, inPileUuid, deleted);
+ static async rerenderTokenHud() {
+ return itemPileSocket.executeForEveryone(SOCKET_HANDLERS.RERENDER_TOKEN_HUD);
}
/**
* @private
*/
- static async _rerenderItemPileInventoryApplication(inPileUuid, deleted = false) {
- return ItemPileInventory.rerenderActiveApp(inPileUuid, deleted);
+ static async _rerenderTokenHud() {
+ if (!canvas.tokens.hud.rendered) return;
+ await canvas.tokens.hud.render(true)
+ return true;
+ }
+
+ /**
+ * Checks whether an item (or item data) is of a type that is not allowed. If an array whether that type is allowed
+ * or not, returning the type if it is NOT allowed.
+ *
+ * @param {Item/Object} item
+ * @param {array/boolean} [itemTypeFilters=false]
+ * @return {boolean/string}
+ */
+ static isItemTypeDisallowed(item, itemTypeFilters = false) {
+ if (!API.ITEM_TYPE_ATTRIBUTE) return false;
+ if (!Array.isArray(itemTypeFilters)) itemTypeFilters = API.ITEM_TYPE_FILTERS;
+ const itemType = getProperty(item, API.ITEM_TYPE_ATTRIBUTE);
+ if (itemTypeFilters.includes(itemType)) {
+ return itemType;
+ }
+ return false;
}
/* -------- PRIVATE ITEM PILE METHODS -------- */
diff --git a/scripts/module.js b/scripts/module.js
index b915765d..00f93b53 100644
--- a/scripts/module.js
+++ b/scripts/module.js
@@ -58,7 +58,6 @@ Hooks.once("ready", () => {
throw lib.custom_error(`Item Piles requires the 'socketlib' module. Please ${word} it.`)
}
checkSystem();
- checkIncompatibilities();
registerHotkeys();
Hooks.callAll(HOOKS.READY);
})
diff --git a/scripts/settings.js b/scripts/settings.js
index a6f22d7c..ddc4c138 100644
--- a/scripts/settings.js
+++ b/scripts/settings.js
@@ -214,21 +214,4 @@ export async function checkSystem(){
applyDefaultSettings();
}
-}
-
-export async function checkIncompatibilities(){
-
-
- if(game.settings.get('monks-active-tiles', 'drop-item') && !game.settings.get(CONSTANTS.MODULE_NAME, 'monksActiveTilesDropItemWarning')){
- await Dialog.prompt({
- title: "Module Incompatibility",
- content: lib.dialogWarning(`The module "Monks Active Tile Triggers" has a setting called "Drop Item on Canvas", which when enabled creates a tile when an item is dropped on the canvas. This clashes with Item Piles, and it is recommended you turn this setting off.`),
- label: "OK",
- callback: () => {
- game.settings.set(CONSTANTS.MODULE_NAME, 'monksActiveTilesDropItemWarning', true)
- }
- });
- }
-
-
}
\ No newline at end of file