Compare commits
27 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| cf3cbb892c | |||
| 2ce7e57f43 | |||
| e20b259d99 | |||
| 13df0ccecd | |||
| 8e565d24ae | |||
| 81f455c63a | |||
| 42529d3df6 | |||
| 9a40e53d2e | |||
| 01857f6ef4 | |||
| 1215ddf55a | |||
| 2eca460637 | |||
| 3461db568e | |||
| 47c5eecd9c | |||
| 0f1975f9ae | |||
| aafa18a2c2 | |||
| 69b6542a18 | |||
| 6b36b62e71 | |||
| f905657e41 | |||
| a2053c7328 | |||
| e17845dd57 | |||
| ff6769f6a4 | |||
| 869fedd128 | |||
| 2a9d7e7acb | |||
| a4d5e6a131 | |||
| 3c632a342d | |||
| 82b495f4c3 | |||
| f67899500e |
51
CHANGELOG.md
51
CHANGELOG.md
@@ -1,12 +1,57 @@
|
|||||||
## v1.1.0
|
## 1.2.9
|
||||||
|
### Feature revival
|
||||||
|
- The "Tint secret doors" feature is back, but will remain disabled by default.
|
||||||
|
|
||||||
|
## 1.2.8
|
||||||
|
### Compatibility
|
||||||
|
- Smart Doors is now compatible with Foundry 0.8.8
|
||||||
|
- Due to API changes inside Foundry, Smart Doors is no longer compatible with Foundry versions older than 0.8.7
|
||||||
|
|
||||||
|
## 1.2.7
|
||||||
|
### Compatibility
|
||||||
|
- Smart Doors is now compatible with Foundry 0.8.5
|
||||||
|
|
||||||
|
### Feature removals
|
||||||
|
- The door icons now have outlines by defualt in Foundry. As a result the "Door Icon Outline" feature was removed.
|
||||||
|
- Secret doors now have a different icon from regular doors in Foundry, making the "Tint Secret Doors" feature redundant. As a result it was removed.
|
||||||
|
|
||||||
|
## 1.2.6
|
||||||
|
### Compatibility
|
||||||
|
- Smart Doors now uses the libwrapper module and as a result is now compatible with the module "FoundryVTT Arms Reach"
|
||||||
|
|
||||||
|
## 1.2.5
|
||||||
|
### New features
|
||||||
|
- Synchronized doors can now be configured to synchronize their secret door status as well
|
||||||
|
|
||||||
|
## v1.2.4
|
||||||
|
### Bugfix
|
||||||
|
- Fixed a race condition that may cause doors to not be properly synchronized across scenes
|
||||||
|
|
||||||
|
## v1.2.3
|
||||||
|
### Other
|
||||||
|
- Smart Doors is now compatible with Arms Reach
|
||||||
|
|
||||||
|
## v1.2.2
|
||||||
|
### Bugfix
|
||||||
|
- Disabled features are now less likely to interfere with other modules, increasing compatibility.
|
||||||
|
- This module can now be used together with the `Arms Reach` module if the `Toggle Secret Doors` feature is disabled in the settings.
|
||||||
|
|
||||||
|
### Other
|
||||||
|
- Warn the user about incompatibility if they use this module together with `Arms Reach` and have incompatible features enabled.
|
||||||
|
|
||||||
|
## v1.2.1
|
||||||
|
### Other
|
||||||
|
- Verified compatibility with 0.7.9
|
||||||
|
|
||||||
|
## v1.2.0
|
||||||
### New features
|
### New features
|
||||||
- Draw outlines around Door Control icons to increase their visibility
|
- Draw outlines around Door Control icons to increase their visibility
|
||||||
- Execute a macro when someone interacts with a door
|
|
||||||
|
|
||||||
### Other
|
### Other
|
||||||
- Secret doors are now tinted black instead of dark grey.
|
- Secret doors are now tinted black instead of dark grey.
|
||||||
- Setting hints will now be shown below the title of the setting (before it was above)
|
|
||||||
|
|
||||||
|
|
||||||
|
## v1.1.0
|
||||||
### New features
|
### New features
|
||||||
- Tint secret doors grey for the GM to differentiate them from regular doors
|
- Tint secret doors grey for the GM to differentiate them from regular doors
|
||||||
- Toggle doors between secret and normal with ctrl+click
|
- Toggle doors between secret and normal with ctrl+click
|
||||||
|
|||||||
21
README.md
21
README.md
@@ -1,3 +1,5 @@
|
|||||||
|
[](https://ko-fi.com/staebchenfisch)
|
||||||
|
|
||||||
# Smart Doors
|
# Smart Doors
|
||||||
Makes doors smarter. Allows doors to synchronize across multiple scenes and sends chat messages when players try to open locked doors (and also tells you which of the doors).
|
Makes doors smarter. Allows doors to synchronize across multiple scenes and sends chat messages when players try to open locked doors (and also tells you which of the doors).
|
||||||
|
|
||||||
@@ -8,21 +10,10 @@ Makes doors smarter. Allows doors to synchronize across multiple scenes and send
|
|||||||
|
|
||||||
Door Control icons will be rendered the same size in every scene, regardless of the configured grid size. The size of the icons is configurable.
|
Door Control icons will be rendered the same size in every scene, regardless of the configured grid size. The size of the icons is configurable.
|
||||||
|
|
||||||
### Door Control Outline
|
|
||||||

|
|
||||||
|
|
||||||
Door Control icons will be rendered with an outline to improve their visibility on bright backgrounds.
|
|
||||||
|
|
||||||
### Tint Secret Doors
|
|
||||||

|
|
||||||
|
|
||||||
Which where the secret doors again? This tints all secret doors grey in the GM view, allowing to easily differentiate between normal and secret doors.
|
|
||||||
|
|
||||||
|
|
||||||
### Toggle Secret Doors
|
### Toggle Secret Doors
|
||||||

|

|
||||||
|
|
||||||
Easily reveal secret doors to players. Strg+left click secrets doors to turn them into regular doors. Strg+left click can also be done on normal doors to turn them into secret doors. Using this in combination with Tint Secret Doors is recommended so you can actually see what you are doing.
|
Easily reveal secret doors to players. Ctrl+left click secrets doors to turn them into regular doors. Ctrl+left click can also be done on normal doors to turn them into secret doors. Using this in combination with Tint Secret Doors is recommended so you can actually see what you are doing.
|
||||||
|
|
||||||
|
|
||||||
### Locked Door Alerts
|
### Locked Door Alerts
|
||||||
@@ -32,6 +23,10 @@ Keep everyone informed who tried to open which door. Whenever a player tries to
|
|||||||
|
|
||||||
If the GM tries to open a locked door the sound will only played for him and no chat message will be sent.
|
If the GM tries to open a locked door the sound will only played for him and no chat message will be sent.
|
||||||
|
|
||||||
|
### Tint Secret Doors
|
||||||
|
This tints secret doors in a gay shade to make them easier to discern from regular doors when being zoomed further out.
|
||||||
|
|
||||||
|
|
||||||
### Synchronized Doors
|
### Synchronized Doors
|
||||||

|

|
||||||
|
|
||||||
@@ -44,7 +39,7 @@ To set up door synchronization, assign all doors that should be synchronized to
|
|||||||
|
|
||||||
Once a Synchronization Group is set up for multiple doors, simply open/close/lock/unlock one of the doors to achieve the same effect on other doors as well.
|
Once a Synchronization Group is set up for multiple doors, simply open/close/lock/unlock one of the doors to achieve the same effect on other doors as well.
|
||||||
|
|
||||||
## Planned Features
|
## Features ideas
|
||||||
- Attach macros to doors that are being executed when the door is being opened/closed
|
- Attach macros to doors that are being executed when the door is being opened/closed
|
||||||
- Give out keys to players, that allow them to lock/unlock associated doors
|
- Give out keys to players, that allow them to lock/unlock associated doors
|
||||||
- Doors that can only be seen from one side when closed
|
- Doors that can only be seen from one side when closed
|
||||||
|
|||||||
26
lang/en.json
26
lang/en.json
@@ -1,10 +1,6 @@
|
|||||||
{
|
{
|
||||||
"smart-doors": {
|
"smart-doors": {
|
||||||
"settings": {
|
"settings": {
|
||||||
"doorControlOutline": {
|
|
||||||
"name": "Door Control Outline",
|
|
||||||
"hint": "Draw outlines around Door Control icons to increase their visiblity"
|
|
||||||
},
|
|
||||||
"doorControlSizeFactor": {
|
"doorControlSizeFactor": {
|
||||||
"name": "Door Control Size Factor",
|
"name": "Door Control Size Factor",
|
||||||
"hint": "Defines by which factor the size of the door control icons should be scaled up"
|
"hint": "Defines by which factor the size of the door control icons should be scaled up"
|
||||||
@@ -17,10 +13,6 @@
|
|||||||
"name": "Locked Door Alert",
|
"name": "Locked Door Alert",
|
||||||
"hint": "Send a message in chat when a player tried to open a locked door"
|
"hint": "Send a message in chat when a player tried to open a locked door"
|
||||||
},
|
},
|
||||||
"macros": {
|
|
||||||
"name": "Door Interaction Macros",
|
|
||||||
"hint": "Trigger a macro when a door is being interacted with"
|
|
||||||
},
|
|
||||||
"synchronizedDoors": {
|
"synchronizedDoors": {
|
||||||
"name": "Synchronized Doors",
|
"name": "Synchronized Doors",
|
||||||
"hint": "Synchronize the state of configured doors"
|
"hint": "Synchronize the state of configured doors"
|
||||||
@@ -31,29 +23,15 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"ui": {
|
"ui": {
|
||||||
"form": {
|
|
||||||
"macroExecuteEverywhere": {
|
|
||||||
"name": "Execute on all clients",
|
|
||||||
"hint": "If disabled the macro will be executed on the GMs client"
|
|
||||||
},
|
|
||||||
"macroName": {
|
|
||||||
"name": "Macro Name",
|
|
||||||
"hint": "The name of the macro that should be executed when this door is interacted with. No macro is executed if left blank."
|
|
||||||
},
|
|
||||||
"macroArguments": {
|
|
||||||
"name": "Marco Parameters",
|
|
||||||
"hint": "The parameters passed to the macro. Any JSON string is valid."
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"messages": {
|
"messages": {
|
||||||
"argsInvalidJson": "The macro arguments must be valid JSON. See console for details.",
|
|
||||||
"migrating": "Migrating Smart Doors to version {version}. Please don't close the application.",
|
"migrating": "Migrating Smart Doors to version {version}. Please don't close the application.",
|
||||||
"migrationDone": "Smart Doors successfully migrated to version {version}.",
|
"migrationDone": "Smart Doors successfully migrated to version {version}.",
|
||||||
"unknownVersion": "Smart Doors migration failed with the error: Unkown Version {version}. Please report this to the Smart Doors issue tracker. To prevent possible data loss don't use this plugin until this error is fixed."
|
"unknownVersion": "Smart Doors migration failed with the error: Unkown Version {version}. Please report this to the Smart Doors issue tracker. To prevent possible data loss don't use this plugin until this error is fixed."
|
||||||
},
|
},
|
||||||
"synchronizedDoors": {
|
"synchronizedDoors": {
|
||||||
"description": "State changes of doors in the same synchronization group will be synchronized across scenes. Leave blank to disable synchronization for this door.",
|
"description": "State changes of doors in the same synchronization group will be synchronized across scenes. Leave blank to disable synchronization for this door.",
|
||||||
"groupName": "Synchronization Group"
|
"groupName": "Synchronization Group",
|
||||||
|
"synchronizeSecretStatus": "Synchronize Secret Status"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
61
lib/libwrapper_shim.js
Normal file
61
lib/libwrapper_shim.js
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
// Copyright © 2021 fvtt-lib-wrapper Rui Pinheiro
|
||||||
|
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
// A shim for the libWrapper library
|
||||||
|
export let libWrapper = undefined;
|
||||||
|
|
||||||
|
Hooks.once('init', () => {
|
||||||
|
// Check if the real module is already loaded - if so, use it
|
||||||
|
if(globalThis.libWrapper && !(globalThis.libWrapper.is_fallback ?? true)) {
|
||||||
|
libWrapper = globalThis.libWrapper;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback implementation
|
||||||
|
libWrapper = class {
|
||||||
|
static get is_fallback() { return true };
|
||||||
|
|
||||||
|
static register(module, target, fn, type="MIXED", {chain=undefined}={}) {
|
||||||
|
const is_setter = target.endsWith('#set');
|
||||||
|
target = !is_setter ? target : target.slice(0, -4);
|
||||||
|
const split = target.split('.');
|
||||||
|
const fn_name = split.pop();
|
||||||
|
const root_nm = split.splice(0,1)[0];
|
||||||
|
const _eval = eval; // The browser doesn't expose all global variables (e.g. 'Game') inside globalThis, but it does to an eval. We copy it to a variable to have it run in global scope.
|
||||||
|
const obj = split.reduce((x,y)=>x[y], globalThis[root_nm] ?? _eval(root_nm));
|
||||||
|
|
||||||
|
let iObj = obj;
|
||||||
|
let descriptor = null;
|
||||||
|
while(iObj) {
|
||||||
|
descriptor = Object.getOwnPropertyDescriptor(iObj, fn_name);
|
||||||
|
if(descriptor) break;
|
||||||
|
iObj = Object.getPrototypeOf(iObj);
|
||||||
|
}
|
||||||
|
if(!descriptor || descriptor?.configurable === false) throw `libWrapper Shim: '${target}' does not exist, could not be found, or has a non-configurable descriptor.`;
|
||||||
|
|
||||||
|
let original = null;
|
||||||
|
const wrapper = (chain ?? type != 'OVERRIDE') ? function() { return fn.call(this, original.bind(this), ...arguments); } : function() { return fn.apply(this, arguments); };
|
||||||
|
|
||||||
|
if(!is_setter) {
|
||||||
|
if(descriptor.value) {
|
||||||
|
original = descriptor.value;
|
||||||
|
descriptor.value = wrapper;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
original = descriptor.get;
|
||||||
|
descriptor.get = wrapper;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if(!descriptor.set) throw `libWrapper Shim: '${target}' does not have a setter`;
|
||||||
|
original = descriptor.set;
|
||||||
|
descriptor.set = wrapper;
|
||||||
|
}
|
||||||
|
|
||||||
|
descriptor.configurable = true;
|
||||||
|
Object.defineProperty(obj, fn_name, descriptor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
@@ -1,152 +0,0 @@
|
|||||||
/*
|
|
||||||
|
|
||||||
This is a modified version of the PIXI outline filter (https://github.com/pixijs/pixi-filters/tree/master/filters/outline)
|
|
||||||
and is licensed under the MIT license.
|
|
||||||
|
|
||||||
The MIT License
|
|
||||||
|
|
||||||
Copyright (c) 2013-2017 Mathew Groves, Chad Engler
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in
|
|
||||||
all copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
THE SOFTWARE.
|
|
||||||
|
|
||||||
*/
|
|
||||||
|
|
||||||
const vertex = `attribute vec2 aVertexPosition;
|
|
||||||
attribute vec2 aTextureCoord;
|
|
||||||
|
|
||||||
uniform mat3 projectionMatrix;
|
|
||||||
|
|
||||||
varying vec2 vTextureCoord;
|
|
||||||
|
|
||||||
void main(void)
|
|
||||||
{
|
|
||||||
gl_Position = vec4((projectionMatrix * vec3(aVertexPosition, 1.0)).xy, 0.0, 1.0);
|
|
||||||
vTextureCoord = aTextureCoord;
|
|
||||||
}`
|
|
||||||
|
|
||||||
const fragment = `varying vec2 vTextureCoord;
|
|
||||||
uniform sampler2D uSampler;
|
|
||||||
|
|
||||||
uniform vec2 thickness;
|
|
||||||
uniform vec4 outlineColor;
|
|
||||||
uniform vec4 filterClamp;
|
|
||||||
|
|
||||||
const float DOUBLE_PI = 3.14159265358979323846264 * 2.;
|
|
||||||
|
|
||||||
void main(void) {
|
|
||||||
vec4 ownColor = texture2D(uSampler, vTextureCoord);
|
|
||||||
vec4 curColor;
|
|
||||||
float maxAlpha = 0.;
|
|
||||||
vec2 displaced;
|
|
||||||
for (float angle = 0.; angle <= DOUBLE_PI; angle += \${angleStep}) {
|
|
||||||
displaced.x = vTextureCoord.x + thickness.x * cos(angle);
|
|
||||||
displaced.y = vTextureCoord.y + thickness.y * sin(angle);
|
|
||||||
curColor = texture2D(uSampler, clamp(displaced, filterClamp.xy, filterClamp.zw));
|
|
||||||
maxAlpha = max(maxAlpha, curColor.a);
|
|
||||||
}
|
|
||||||
float resultAlpha = max(maxAlpha, ownColor.a);
|
|
||||||
// Original line:
|
|
||||||
// gl_FragColor = vec4((ownColor.rgb + outlineColor.rgb * (1. - ownColor.a)) * resultAlpha, resultAlpha);
|
|
||||||
gl_FragColor = vec4(ownColor.rgb + outlineColor.rgb * (resultAlpha - ownColor.a), resultAlpha);
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
/**
|
|
||||||
* OutlineFilter, originally by mishaa
|
|
||||||
* http://www.html5gamedevs.com/topic/10640-outline-a-sprite-change-certain-colors/?p=69966
|
|
||||||
* http://codepen.io/mishaa/pen/emGNRB<br>
|
|
||||||
* 
|
|
||||||
*
|
|
||||||
* @class
|
|
||||||
* @extends PIXI.Filter
|
|
||||||
* @memberof PIXI.filters
|
|
||||||
* @see {@link https://www.npmjs.com/package/@pixi/filter-outline|@pixi/filter-outline}
|
|
||||||
* @see {@link https://www.npmjs.com/package/pixi-filters|pixi-filters}
|
|
||||||
* @param {number} [thickness=1] The tickness of the outline. Make it 2 times more for resolution 2
|
|
||||||
* @param {number} [color=0x000000] The color of the outline.
|
|
||||||
* @param {number} [quality=0.1] The quality of the outline from `0` to `1`, using a higher quality
|
|
||||||
* setting will result in slower performance and more accuracy.
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* someSprite.filters = [new OutlineFilter(2, 0x99ff99)];
|
|
||||||
*/
|
|
||||||
class OutlineFilter extends PIXI.Filter {
|
|
||||||
|
|
||||||
constructor(thickness = 1, color = 0x000000, quality = 0.1) {
|
|
||||||
const samples = Math.max(
|
|
||||||
quality * OutlineFilter.MAX_SAMPLES,
|
|
||||||
OutlineFilter.MIN_SAMPLES
|
|
||||||
);
|
|
||||||
const angleStep = (Math.PI * 2 / samples).toFixed(7);
|
|
||||||
|
|
||||||
super(vertex, fragment.replace(/\$\{angleStep\}/, angleStep));
|
|
||||||
this.uniforms.thickness = new Float32Array([0, 0]);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The thickness of the outline.
|
|
||||||
* @member {number}
|
|
||||||
* @default 1
|
|
||||||
*/
|
|
||||||
this.thickness = thickness;
|
|
||||||
|
|
||||||
this.uniforms.outlineColor = new Float32Array([0, 0, 0, 1]);
|
|
||||||
this.color = color;
|
|
||||||
|
|
||||||
this.quality = quality;
|
|
||||||
}
|
|
||||||
|
|
||||||
apply(filterManager, input, output, clear) {
|
|
||||||
this.uniforms.thickness[0] = this.thickness / input._frame.width;
|
|
||||||
this.uniforms.thickness[1] = this.thickness / input._frame.height;
|
|
||||||
|
|
||||||
filterManager.applyFilter(this, input, output, clear);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The color of the glow.
|
|
||||||
* @member {number}
|
|
||||||
* @default 0x000000
|
|
||||||
*/
|
|
||||||
get color() {
|
|
||||||
return PIXI.utils.rgb2hex(this.uniforms.outlineColor);
|
|
||||||
}
|
|
||||||
set color(value) {
|
|
||||||
PIXI.utils.hex2rgb(value, this.uniforms.outlineColor);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The minimum number of samples for rendering outline.
|
|
||||||
* @static
|
|
||||||
* @member {number} MIN_SAMPLES
|
|
||||||
* @memberof PIXI.filters.OutlineFilter
|
|
||||||
* @default 1
|
|
||||||
*/
|
|
||||||
OutlineFilter.MIN_SAMPLES = 1;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The maximum number of samples for rendering outline.
|
|
||||||
* @static
|
|
||||||
* @member {number} MAX_SAMPLES
|
|
||||||
* @memberof PIXI.filters.OutlineFilter
|
|
||||||
* @default 100
|
|
||||||
*/
|
|
||||||
OutlineFilter.MAX_SAMPLES = 100;
|
|
||||||
|
|
||||||
export { OutlineFilter };
|
|
||||||
20
module.json
20
module.json
@@ -2,11 +2,18 @@
|
|||||||
"name": "smart-doors",
|
"name": "smart-doors",
|
||||||
"title": "Smart Doors",
|
"title": "Smart Doors",
|
||||||
"description": "Makes doors smarter. Allows doors to synchronize across multiple scenes and sends chat messages when players try to open locked doors.",
|
"description": "Makes doors smarter. Allows doors to synchronize across multiple scenes and sends chat messages when players try to open locked doors.",
|
||||||
"version": "1.1.0",
|
"version": "1.2.9",
|
||||||
"minimumCoreVersion" : "0.7.7",
|
"minimumCoreVersion" : "0.8.7",
|
||||||
"compatibleCoreVersion" : "0.7.8",
|
"compatibleCoreVersion" : "0.8.8",
|
||||||
"author": "Manuel Vögele",
|
"authors": [
|
||||||
|
{
|
||||||
|
"name": "Manuel Vögele",
|
||||||
|
"email": "develop@manuel-voegele.de",
|
||||||
|
"discord": "Stäbchenfisch#5107"
|
||||||
|
}
|
||||||
|
],
|
||||||
"esmodules": [
|
"esmodules": [
|
||||||
|
"lib/libwrapper_shim.js",
|
||||||
"src/main.js"
|
"src/main.js"
|
||||||
],
|
],
|
||||||
"languages": [
|
"languages": [
|
||||||
@@ -18,8 +25,9 @@
|
|||||||
],
|
],
|
||||||
"url": "https://github.com/manuelVo/foundryvtt-smart-doors",
|
"url": "https://github.com/manuelVo/foundryvtt-smart-doors",
|
||||||
"manifest": "https://raw.githubusercontent.com/manuelVo/foundryvtt-smart-doors/master/module.json",
|
"manifest": "https://raw.githubusercontent.com/manuelVo/foundryvtt-smart-doors/master/module.json",
|
||||||
"download": "https://github.com/manuelVo/foundryvtt-smart-doors/archive/v1.1.0.zip",
|
"download": "https://github.com/manuelVo/foundryvtt-smart-doors/archive/v1.2.9.zip",
|
||||||
"readme": "https://github.com/manuelVo/foundryvtt-smart-doors/blob/master/README.md",
|
"readme": "https://github.com/manuelVo/foundryvtt-smart-doors/blob/master/README.md",
|
||||||
"changelog": "https://github.com/manuelVo/foundryvtt-smart-doors/blob/master/CHANGELOG.md",
|
"changelog": "https://github.com/manuelVo/foundryvtt-smart-doors/blob/master/CHANGELOG.md",
|
||||||
"bugs": "https://github.com/manuelVo/foundryvtt-smart-doors/issues"
|
"bugs": "https://github.com/manuelVo/foundryvtt-smart-doors/issues",
|
||||||
|
"allowBugReporter": true
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,13 +1,14 @@
|
|||||||
|
import { libWrapper } from "../../lib/libwrapper_shim.js"
|
||||||
import {settingsKey} from "../settings.js"
|
import {settingsKey} from "../settings.js"
|
||||||
|
|
||||||
// Adjust the repositioning formula for the door controls
|
// Adjust the repositioning formula for the door controls
|
||||||
export function hookDoorControlReposition() {
|
export function hookDoorControlReposition() {
|
||||||
DoorControl.prototype.reposition = function () {
|
libWrapper.register("smart-doors", "DoorControl.prototype.reposition", function () {
|
||||||
let gridSize = this.wall.scene.data.grid
|
let gridSize = this.wall.scene.data.grid
|
||||||
gridSize *= game.settings.get(settingsKey, "doorControlSizeFactor")
|
gridSize *= game.settings.get(settingsKey, "doorControlSizeFactor")
|
||||||
const pos = this.wall.midpoint.map(p => p - gridSize * 0.2)
|
const pos = this.wall.midpoint.map(p => p - gridSize * 0.2)
|
||||||
this.position.set(...pos)
|
this.position.set(...pos)
|
||||||
}
|
}, "OVERRIDE");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set the size of all door controls in relation to the grid size so it'll have a constant percieved size
|
// Set the size of all door controls in relation to the grid size so it'll have a constant percieved size
|
||||||
|
|||||||
@@ -1,24 +0,0 @@
|
|||||||
import {settingsKey} from "../settings.js"
|
|
||||||
import {OutlineFilter} from "../../lib/outline_filter/outline_filter.js"
|
|
||||||
|
|
||||||
export function onDoorControlPostDraw() {
|
|
||||||
if (!game.settings.get(settingsKey, "doorControlOutline"))
|
|
||||||
return
|
|
||||||
|
|
||||||
const types = CONST.WALL_DOOR_TYPES
|
|
||||||
if (this.wall.data.door === types.NONE)
|
|
||||||
return
|
|
||||||
|
|
||||||
// Remove all OutlineFilters from current filters
|
|
||||||
let pixiFilters = this.icon.filters || []
|
|
||||||
pixiFilters = pixiFilters.filter(pixiFilter => !(pixiFilter instanceof OutlineFilter))
|
|
||||||
|
|
||||||
let outlineFilter;
|
|
||||||
if (this.wall.data.door === types.SECRET && game.settings.get(settingsKey, "highlightSecretDoors"))
|
|
||||||
outlineFilter = new OutlineFilter(1, 0xFFFFFF)
|
|
||||||
else
|
|
||||||
outlineFilter = new OutlineFilter(1, 0x000000)
|
|
||||||
|
|
||||||
pixiFilters.push(outlineFilter)
|
|
||||||
this.icon.filters = pixiFilters
|
|
||||||
}
|
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
import {settingsKey} from "../settings.js"
|
|
||||||
import {textInput, checkboxInput, injectSettings} from "../form.js"
|
|
||||||
|
|
||||||
// Inject settings for synchronized doors
|
|
||||||
export function onRederWallConfig(wallConfig, html, data) {
|
|
||||||
if (data.isDoor && game.settings.get(settingsKey, "macros")) {
|
|
||||||
|
|
||||||
const settings = [
|
|
||||||
textInput("macroName", data.object.flags.smartdoors?.macro?.name),
|
|
||||||
textInput("macroArguments", JSON.stringify(data.object.flags.smartdoors?.macro?.args ?? undefined)),
|
|
||||||
checkboxInput("macroExecuteEverywhere", data.object.flags.smartdoors?.macro?.executeEverywhere),
|
|
||||||
]
|
|
||||||
|
|
||||||
injectSettings(html, settings)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check data input by the user for validity
|
|
||||||
export async function onWallConfigPreUpdate(event, formData) {
|
|
||||||
const args = formData.macroArguments || "null"
|
|
||||||
|
|
||||||
try {
|
|
||||||
// Check if args can be converted to JSON
|
|
||||||
JSON.parse(args)
|
|
||||||
}
|
|
||||||
catch (error) {
|
|
||||||
ui.notifications.error(game.i18n.localize("smart-doors.ui.messages.argsInvalidJson"))
|
|
||||||
// Rethrow the error to stop the update and prevent the dialog from closing
|
|
||||||
throw(error)
|
|
||||||
}
|
|
||||||
|
|
||||||
// The JSON is valid. Assign "null" instead of an empty string if necessary
|
|
||||||
formData.macroArguments = args
|
|
||||||
}
|
|
||||||
|
|
||||||
// Store our custom data from the WallConfig dialog
|
|
||||||
export async function onWallConfigUpdate(event, formData) {
|
|
||||||
let ids = this.options.editTargets;
|
|
||||||
if (ids.length == 0) {
|
|
||||||
ids = [this.object.data._id];
|
|
||||||
}
|
|
||||||
|
|
||||||
const updateData = {flags: {smartdoors: {macro: {
|
|
||||||
name: formData.macroName,
|
|
||||||
args: JSON.parse(formData.macroArguments),
|
|
||||||
executeEverywhere: formData.macroExecuteEverywhere
|
|
||||||
}}}}
|
|
||||||
|
|
||||||
// Update all the edited walls
|
|
||||||
const updateDataset = ids.reduce((dataset, id) => {
|
|
||||||
dataset.push({_id: id, ...updateData})
|
|
||||||
return dataset
|
|
||||||
}, [])
|
|
||||||
return canvas.scene.updateEmbeddedEntity("Wall", updateDataset)
|
|
||||||
}
|
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import {settingsKey} from "../settings.js"
|
import {settingsKey} from "../settings.js"
|
||||||
|
|
||||||
const SECRET_DOOR_TINT = 0x000000
|
const SECRET_DOOR_TINT = 0x888888
|
||||||
|
|
||||||
// Tint all secret doors dark grey
|
// Tint all secret doors dark grey
|
||||||
export function onCanvasReady(currentCanvas) {
|
export function onCanvasReady(currentCanvas) {
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ export function onRenderChatMessage(message, html, data) {
|
|||||||
|
|
||||||
// Tint on mouse enter
|
// Tint on mouse enter
|
||||||
const mouseEnter = function () {
|
const mouseEnter = function () {
|
||||||
const sourceDoor = canvas.controls.doors.children.find(door => door.wall.data._id === source.wall && door.wall.scene.id === source.scene);
|
const sourceDoor = canvas.controls.doors.children.find(door => door.wall.id === source.wall && door.wall.scene.id === source.scene);
|
||||||
if (sourceDoor)
|
if (sourceDoor)
|
||||||
sourceDoor.icon.tint = 0xff0000;
|
sourceDoor.icon.tint = 0xff0000;
|
||||||
}
|
}
|
||||||
@@ -17,7 +17,7 @@ export function onRenderChatMessage(message, html, data) {
|
|||||||
|
|
||||||
// Remove tint on mouse leave
|
// Remove tint on mouse leave
|
||||||
const mouseLeave = function () {
|
const mouseLeave = function () {
|
||||||
const sourceDoor = canvas.controls.doors.children.find(door => door.wall.data._id === source.wall && door.wall.scene.id === source.scene);
|
const sourceDoor = canvas.controls.doors.children.find(door => door.wall.id === source.wall && door.wall.scene.id === source.scene);
|
||||||
if (sourceDoor)
|
if (sourceDoor)
|
||||||
sourceDoor.icon.tint = 0xffffff;
|
sourceDoor.icon.tint = 0xffffff;
|
||||||
}
|
}
|
||||||
@@ -26,15 +26,15 @@ export function onRenderChatMessage(message, html, data) {
|
|||||||
|
|
||||||
// Creates a chat message stating that a player tried to open a locked door
|
// Creates a chat message stating that a player tried to open a locked door
|
||||||
export function onDoorLeftClick() {
|
export function onDoorLeftClick() {
|
||||||
const state = this.wall.data.ds
|
|
||||||
const states = CONST.WALL_DOOR_STATES
|
|
||||||
|
|
||||||
// Check if this feature is enabled
|
// Check if this feature is enabled
|
||||||
if (!game.settings.get(settingsKey, "lockedDoorAlert"))
|
if (!game.settings.get(settingsKey, "lockedDoorAlert"))
|
||||||
return false
|
return false
|
||||||
|
|
||||||
|
const state = this.wall.data.ds
|
||||||
|
const states = CONST.WALL_DOOR_STATES
|
||||||
|
|
||||||
// Only create messages when the door is locked.
|
// Only create messages when the door is locked.
|
||||||
if (state != states.LOCKED)
|
if (state !== states.LOCKED)
|
||||||
return false
|
return false
|
||||||
|
|
||||||
// Generate no message if the gm attempts to open the door
|
// Generate no message if the gm attempts to open the door
|
||||||
@@ -43,7 +43,7 @@ export function onDoorLeftClick() {
|
|||||||
|
|
||||||
// Create and send the chat message
|
// Create and send the chat message
|
||||||
const message = {}
|
const message = {}
|
||||||
message.user = game.user
|
message.user = game.user.id;
|
||||||
if (game.user.character)
|
if (game.user.character)
|
||||||
message.speaker = {actor: game.user.character}
|
message.speaker = {actor: game.user.character}
|
||||||
message.content = "Just tried to open a locked door"
|
message.content = "Just tried to open a locked door"
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import * as Util from "../util.js"
|
|||||||
|
|
||||||
// Inject settings for synchronized doors
|
// Inject settings for synchronized doors
|
||||||
export function onRederWallConfig(wallConfig, html, data) {
|
export function onRederWallConfig(wallConfig, html, data) {
|
||||||
if (data.isDoor && game.settings.get(settingsKey, "synchronizedDoors")) {
|
if (game.settings.get(settingsKey, "synchronizedDoors") && data.isDoor) {
|
||||||
// Inject settings
|
// Inject settings
|
||||||
const synchronizedSettings = `
|
const synchronizedSettings = `
|
||||||
<p class="notes">${game.i18n.localize("smart-doors.ui.synchronizedDoors.description")}</p>
|
<p class="notes">${game.i18n.localize("smart-doors.ui.synchronizedDoors.description")}</p>
|
||||||
@@ -11,49 +11,71 @@ export function onRederWallConfig(wallConfig, html, data) {
|
|||||||
<label for="synchronizationGroup">${game.i18n.localize("smart-doors.ui.synchronizedDoors.groupName")}</label>
|
<label for="synchronizationGroup">${game.i18n.localize("smart-doors.ui.synchronizedDoors.groupName")}</label>
|
||||||
<input type="text" name="synchronizationGroup"/>
|
<input type="text" name="synchronizationGroup"/>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="synchronizeSecretStatus">${game.i18n.localize("smart-doors.ui.synchronizedDoors.synchronizeSecretStatus")}</label>
|
||||||
|
<input type="checkbox" name="synchronizeSecretStatus" value="true"/>
|
||||||
|
</div>
|
||||||
`
|
`
|
||||||
html.find(".form-group").last().after(synchronizedSettings)
|
html.find(".form-group").last().after(synchronizedSettings)
|
||||||
|
|
||||||
const smartdoorsData = data.object.flags.smartdoors
|
const smartdoorsData = data.object.flags.smartdoors
|
||||||
// Fill the injected input fields with values
|
// Fill the injected input fields with values
|
||||||
const input = (name) => html.find(`input[name="${name}"]`)
|
const input = (name) => html.find(`input[name="${name}"]`); // input is a helper function to search for a input field by it's name
|
||||||
input("synchronizationGroup").prop("value", smartdoorsData?.synchronizationGroup)
|
input("synchronizationGroup").prop("value", smartdoorsData?.synchronizationGroup)
|
||||||
|
input("synchronizeSecretStatus").prop("checked", smartdoorsData?.synchronizeSecretStatus);
|
||||||
|
|
||||||
|
// Recalculate config window height
|
||||||
|
wallConfig.setPosition({height: "auto"})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Store our custom data from the WallConfig dialog
|
// Store our custom data from the WallConfig dialog
|
||||||
export async function onWallConfigUpdate(event, formData) {
|
export async function onWallConfigUpdate(event, formData) {
|
||||||
const updateData = {flags: {smartdoors: {synchronizationGroup: formData.synchronizationGroup}}}
|
const synchronizeSecretStatus = formData.synchronizeSecretStatus;
|
||||||
let ids = this.options.editTargets;
|
const updateData = {flags: {smartdoors: {synchronizationGroup: formData.synchronizationGroup}}};
|
||||||
|
let ids = this.editTargets;
|
||||||
if (ids.length == 0) {
|
if (ids.length == 0) {
|
||||||
ids = [this.object.data._id];
|
ids = [this.object.id];
|
||||||
}
|
}
|
||||||
|
|
||||||
// If a synchronization group is set, get the state of existing doors and assume their state
|
// If a synchronization group is set, get the state of existing doors and assume their state
|
||||||
if (formData.synchronizationGroup) {
|
if (formData.synchronizationGroup) {
|
||||||
|
// Update the synchronizeSecretStatus flag
|
||||||
|
updateData.flags.smartdoors.synchronizeSecretStatus = synchronizeSecretStatus;
|
||||||
|
|
||||||
// Search for other doors in the synchronization group that aren't in the list of edited doors
|
// Search for other doors in the synchronization group that aren't in the list of edited doors
|
||||||
const doorInGroup = Util.findInAllWalls(wall => {
|
const doorInGroup = Util.findInAllWalls(wall => {
|
||||||
// We only search for doors
|
// We only search for doors
|
||||||
if (!wall.door)
|
if (!wall.data.door)
|
||||||
return false
|
return false
|
||||||
// We only want doors in the same synchronization group
|
// We only want doors in the same synchronization group
|
||||||
if (wall.flags.smartdoors?.synchronizationGroup !== formData.synchronizationGroup)
|
if (wall.data.flags.smartdoors?.synchronizationGroup !== formData.synchronizationGroup)
|
||||||
return false
|
return false
|
||||||
// Doors on this scene that have their id included in `ids` are currently being changed. Ignore them.
|
// Doors on this scene that have their id included in `ids` are currently being changed. Ignore them.
|
||||||
if (wall.scene === canvas.scene && ids.includes(wall._id))
|
if (wall.parent.id === canvas.scene.id && ids.includes(wall.id))
|
||||||
return false
|
return false
|
||||||
return true
|
return true
|
||||||
})
|
})
|
||||||
if (doorInGroup)
|
if (doorInGroup) {
|
||||||
updateData.ds = doorInGroup.ds;
|
// ds is the door sate in foundry
|
||||||
|
updateData.ds = doorInGroup.data.ds;
|
||||||
|
|
||||||
|
if (synchronizeSecretStatus) {
|
||||||
|
// door is the door type in foundry
|
||||||
|
updateData.door = doorInGroup.data.door;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update all the edited walls
|
// Update all the edited walls
|
||||||
const updateDataset = ids.reduce((dataset, id) => {
|
const updateDataset = ids.map(id => {return {_id: id, ...updateData}});
|
||||||
dataset.push({_id: id, ...updateData})
|
const updateResult = await canvas.scene.updateEmbeddedDocuments("Wall", updateDataset);
|
||||||
return dataset
|
|
||||||
}, [])
|
// If door is synchronized, synchronize secret status among synchronized doors
|
||||||
return canvas.scene.updateEmbeddedEntity("Wall", updateDataset)
|
if (formData.synchronizationGroup)
|
||||||
|
await updateSynchronizedDoors(updateData, formData.synchronizationGroup);
|
||||||
|
|
||||||
|
return updateResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update the state of all synchronized doors
|
// Update the state of all synchronized doors
|
||||||
@@ -118,12 +140,10 @@ export function onDoorRightClick() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Updates all doors in the specified synchronization group with the provided data
|
// Updates all doors in the specified synchronization group with the provided data
|
||||||
function updateSynchronizedDoors(updateData, synchronizationGroup) {
|
export function updateSynchronizedDoors(updateData, synchronizationGroup) {
|
||||||
// Search for doors belonging to the synchronization group in all scenes
|
// Search for doors belonging to the synchronization group in all scenes
|
||||||
let scenes = Util.filterAllWalls(wall => wall.door && wall.flags.smartdoors?.synchronizationGroup === synchronizationGroup);
|
let scenes = Util.filterAllWalls(wall => wall.data.door && wall.data.flags.smartdoors?.synchronizationGroup === synchronizationGroup);
|
||||||
|
|
||||||
// Update all doors in the synchronization group
|
// Update all doors in the synchronization group
|
||||||
scenes.forEach((scene) => {
|
return Promise.all(scenes.map(scene => scene.scene.updateEmbeddedDocuments("Wall", scene.walls.map((wall) => {return {_id: wall.id, ...updateData}}))));
|
||||||
scene.scene.updateEmbeddedEntity("Wall", scene.walls.map((wall) => {return {_id: wall._id, ...updateData}}))
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,11 +1,19 @@
|
|||||||
import {settingsKey} from "../settings.js"
|
import {settingsKey} from "../settings.js"
|
||||||
|
import {updateSynchronizedDoors} from "./synchronized_doors.js";
|
||||||
|
|
||||||
// Toggles between normal and secret doors
|
// Toggles between normal and secret doors
|
||||||
export function onDoorLeftClick(event) {
|
export function onDoorLeftClick(event) {
|
||||||
if (event.data.originalEvent.ctrlKey && game.user.isGM && game.settings.get(settingsKey, "toggleSecretDoors")) {
|
// We don't trust the event to be filled with the expected data for compatibilty with arms reach (which passes a broken event)
|
||||||
|
if (game.settings.get(settingsKey, "toggleSecretDoors") && event.data?.originalEvent?.ctrlKey && game.user.isGM) {
|
||||||
const types = CONST.WALL_DOOR_TYPES
|
const types = CONST.WALL_DOOR_TYPES
|
||||||
const newtype = this.wall.data.door === types.DOOR ? types.SECRET : types.DOOR
|
const newtype = this.wall.data.door === types.DOOR ? types.SECRET : types.DOOR
|
||||||
this.wall.update({door: newtype})
|
const updateData = {door: newtype}
|
||||||
|
const synchronizationGroup = this.wall.data.flags.smartdoors?.synchronizationGroup
|
||||||
|
if (game.settings.get(settingsKey, "synchronizedDoors") && synchronizationGroup && this.wall.data.flags.smartdoors?.synchronizeSecretStatus)
|
||||||
|
updateSynchronizedDoors(updateData, synchronizationGroup)
|
||||||
|
else
|
||||||
|
this.wall.document.update(updateData)
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
|
|||||||
38
src/form.js
38
src/form.js
@@ -1,38 +0,0 @@
|
|||||||
function formEntry(name, input) {
|
|
||||||
return `
|
|
||||||
<div class="form-group">
|
|
||||||
<label for="${name}">${game.i18n.localize(`smart-doors.ui.form.${name}.name`)}</label>
|
|
||||||
${input}
|
|
||||||
</div>
|
|
||||||
<p class="notes">${game.i18n.localize(`smart-doors.ui.form.${name}.hint`)}</p>
|
|
||||||
`
|
|
||||||
}
|
|
||||||
|
|
||||||
export function injectSettings(html, settings) {
|
|
||||||
html.find(".form-group").last().after(settings.join(""))
|
|
||||||
}
|
|
||||||
|
|
||||||
export function textInput(name, value) {
|
|
||||||
return formEntry(name, `<input type="text" name="${escapeHtml(name)}" value="${escapeHtml(value ?? "")}"/>`)
|
|
||||||
}
|
|
||||||
|
|
||||||
export function selectInput(name, values) {
|
|
||||||
// TODO Set selected option
|
|
||||||
let html = `<select name="${name}">`
|
|
||||||
html += values.reduce((html, value) => html + `<option value="${escapeHtml(value)}">${game.i18n.localize(`smart-doors.ui.form.${name}.options.${value}`)}</option>`, "")
|
|
||||||
html += "</select>"
|
|
||||||
return formEntry(name, html)
|
|
||||||
}
|
|
||||||
|
|
||||||
export function checkboxInput(name, checked) {
|
|
||||||
return formEntry(name, `<input type="checkbox" name="${escapeHtml(name)}" value="true" ${checked ? "checked" : ""}/>`)
|
|
||||||
}
|
|
||||||
|
|
||||||
function escapeHtml(unsafe) {
|
|
||||||
return unsafe
|
|
||||||
.replace(/&/g, "&")
|
|
||||||
.replace(/</g, "<")
|
|
||||||
.replace(/>/g, ">")
|
|
||||||
.replace(/"/g, """)
|
|
||||||
.replace(/'/g, "'");
|
|
||||||
}
|
|
||||||
53
src/main.js
53
src/main.js
@@ -1,15 +1,14 @@
|
|||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
|
import {libWrapper} from "../lib/libwrapper_shim.js";
|
||||||
import * as DoorControlIconScale from "./features/door_control_icon_scale.js"
|
import * as DoorControlIconScale from "./features/door_control_icon_scale.js"
|
||||||
import * as DoorControlOutline from "./features/door_control_outline.js"
|
|
||||||
import * as ExecuteMacro from "./features/execute_macro.js"
|
|
||||||
import * as HighlightSecretDoors from "./features/highlight_secret_doors.js"
|
import * as HighlightSecretDoors from "./features/highlight_secret_doors.js"
|
||||||
import * as LockedDoorAlert from "./features/locked_door_alert.js"
|
import * as LockedDoorAlert from "./features/locked_door_alert.js"
|
||||||
import * as SynchronizedDoors from "./features/synchronized_doors.js"
|
import * as SynchronizedDoors from "./features/synchronized_doors.js"
|
||||||
import * as ToggleSecretDoor from "./features/toggle_secret_door.js"
|
import * as ToggleSecretDoor from "./features/toggle_secret_door.js"
|
||||||
|
|
||||||
import {performMigrations} from "./migration.js"
|
import {performMigrations} from "./migration.js"
|
||||||
import {registerSettings} from "./settings.js"
|
import {registerSettings, settingsKey} from "./settings.js"
|
||||||
|
|
||||||
Hooks.once("init", () => {
|
Hooks.once("init", () => {
|
||||||
registerSettings()
|
registerSettings()
|
||||||
@@ -31,63 +30,45 @@ Hooks.on("canvasReady", HighlightSecretDoors.onCanvasReady)
|
|||||||
Hooks.on("updateWall", HighlightSecretDoors.onUpdateWall)
|
Hooks.on("updateWall", HighlightSecretDoors.onUpdateWall)
|
||||||
|
|
||||||
// Inject our custom settings into the WallConfig dialog
|
// Inject our custom settings into the WallConfig dialog
|
||||||
Hooks.on("renderWallConfig", (wallConfig, html, data) => {
|
Hooks.on("renderWallConfig", SynchronizedDoors.onRederWallConfig)
|
||||||
SynchronizedDoors.onRederWallConfig(wallConfig, html, data)
|
|
||||||
ExecuteMacro.onRederWallConfig(wallConfig, html, data)
|
|
||||||
|
|
||||||
// Recalculate config window position and height
|
|
||||||
wallConfig.element[0].style.top = "" // This forces foundry to re-calculate the top position
|
|
||||||
wallConfig.setPosition({height: "auto"})
|
|
||||||
})
|
|
||||||
|
|
||||||
// Hook the update function of the WallConfig dialog so we can store our custom data
|
// Hook the update function of the WallConfig dialog so we can store our custom data
|
||||||
function hookWallConfigUpdate() {
|
function hookWallConfigUpdate() {
|
||||||
// Replace the original function with our custom one
|
// Replace the original function with our custom one
|
||||||
const originalHandler = WallConfig.prototype._updateObject;
|
libWrapper.register("smart-doors", "WallConfig.prototype._updateObject", async function (wrapped, event, formData) {
|
||||||
WallConfig.prototype._updateObject = async function (event, formData) {
|
await wrapped(event, formData);
|
||||||
await ExecuteMacro.onWallConfigPreUpdate.call(this, event, formData)
|
return SynchronizedDoors.onWallConfigUpdate.call(this, event, formData)
|
||||||
|
}, "WRAPPER");
|
||||||
await originalHandler.call(this, event, formData)
|
|
||||||
|
|
||||||
return Promise.all([
|
|
||||||
SynchronizedDoors.onWallConfigUpdate.call(this, event, formData),
|
|
||||||
ExecuteMacro.onWallConfigUpdate.call(this, event, formData),
|
|
||||||
])
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function hookDoorControlDraw() {
|
function hookDoorControlDraw() {
|
||||||
const originalHandler = DoorControl.prototype.draw
|
libWrapper.register("smart-doors", "DoorControl.prototype.draw", async function (wrapped) {
|
||||||
DoorControl.prototype.draw = async function () {
|
const result = await wrapped();
|
||||||
const result = await originalHandler.call(this)
|
|
||||||
DoorControlIconScale.onDoorControlPostDraw.call(this)
|
DoorControlIconScale.onDoorControlPostDraw.call(this)
|
||||||
DoorControlOutline.onDoorControlPostDraw.call(this)
|
return result;
|
||||||
return result
|
}, "WRAPPER");
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hook mouse events on DoorControls to perform our logic.
|
// Hook mouse events on DoorControls to perform our logic.
|
||||||
// If we successfully handled the event block the original handler. Forward the event otherwise.
|
// If we successfully handled the event block the original handler. Forward the event otherwise.
|
||||||
function hookDoorEvents() {
|
function hookDoorEvents() {
|
||||||
// Replace the original mousedown handler with our custom one
|
// Replace the original mousedown handler with our custom one
|
||||||
const originalMouseDownHandler = DoorControl.prototype._onMouseDown
|
libWrapper.register("smart-doors", "DoorControl.prototype._onMouseDown", function (wrapped, event) {
|
||||||
DoorControl.prototype._onMouseDown = function (event) {
|
|
||||||
// Call our handler first. Only allow the original handler to run if our handler returns true
|
// Call our handler first. Only allow the original handler to run if our handler returns true
|
||||||
const eventHandled = onDoorMouseDown.call(this, event)
|
const eventHandled = onDoorMouseDown.call(this, event)
|
||||||
if (eventHandled)
|
if (eventHandled)
|
||||||
return
|
return
|
||||||
return originalMouseDownHandler.call(this, event)
|
return wrapped(event);
|
||||||
}
|
}, "MIXED");
|
||||||
|
|
||||||
// Replace the original rightdown handler with our custom one
|
// Replace the original rightdown handler with our custom one
|
||||||
const originalRightDownHandler = DoorControl.prototype._onRightDown
|
libWrapper.register("smart-doors", "DoorControl.prototype._onRightDown", function (wrapped, event) {
|
||||||
DoorControl.prototype._onRightDown = function (event) {
|
|
||||||
// Call our handler first. Only allow the original handler to run if our handler returns true
|
// Call our handler first. Only allow the original handler to run if our handler returns true
|
||||||
const eventHandled = onDoorRightDown.call(this, event)
|
const eventHandled = onDoorRightDown.call(this, event)
|
||||||
if (eventHandled)
|
if (eventHandled)
|
||||||
return
|
return
|
||||||
return originalRightDownHandler.call(this, event)
|
return wrapped(event);
|
||||||
}
|
}, "MIXED");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Our custom handler for mousedown events on doors
|
// Our custom handler for mousedown events on doors
|
||||||
|
|||||||
@@ -20,9 +20,9 @@ export function performMigrations() {
|
|||||||
// Make a dictionary that maps all door ids to their scenes
|
// Make a dictionary that maps all door ids to their scenes
|
||||||
const walls = game.scenes.reduce((dict, scene) => {
|
const walls = game.scenes.reduce((dict, scene) => {
|
||||||
scene.data.walls.forEach(wall => {
|
scene.data.walls.forEach(wall => {
|
||||||
if (!wall.door)
|
if (!wall.data.door)
|
||||||
return
|
return
|
||||||
dict[wall._id] = scene.id
|
dict[wall.id] = scene.id;
|
||||||
})
|
})
|
||||||
return dict
|
return dict
|
||||||
}, {})
|
}, {})
|
||||||
|
|||||||
@@ -21,22 +21,13 @@ export function registerSettings() {
|
|||||||
default: 1.5,
|
default: 1.5,
|
||||||
onChange: () => location.reload()
|
onChange: () => location.reload()
|
||||||
})
|
})
|
||||||
game.settings.register(settingsKey, "doorControlOutline", {
|
|
||||||
name: "smart-doors.settings.doorControlOutline.name",
|
|
||||||
hint: "smart-doors.settings.doorControlOutline.hint",
|
|
||||||
scope: "client",
|
|
||||||
config: true,
|
|
||||||
type: Boolean,
|
|
||||||
default: true,
|
|
||||||
onChange: () => location.reload(),
|
|
||||||
})
|
|
||||||
game.settings.register(settingsKey, "highlightSecretDoors", {
|
game.settings.register(settingsKey, "highlightSecretDoors", {
|
||||||
name: "smart-doors.settings.highlightSecretDoors.name",
|
name: "smart-doors.settings.highlightSecretDoors.name",
|
||||||
hint: "smart-doors.settings.highlightSecretDoors.hint",
|
hint: "smart-doors.settings.highlightSecretDoors.hint",
|
||||||
scope: "world",
|
scope: "world",
|
||||||
config: true,
|
config: true,
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: true,
|
default: false,
|
||||||
onChange: reloadGM,
|
onChange: reloadGM,
|
||||||
})
|
})
|
||||||
game.settings.register(settingsKey, "toggleSecretDoors", {
|
game.settings.register(settingsKey, "toggleSecretDoors", {
|
||||||
@@ -63,12 +54,4 @@ export function registerSettings() {
|
|||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: true,
|
default: true,
|
||||||
})
|
})
|
||||||
game.settings.register(settingsKey, "macros", {
|
|
||||||
name: "smart-doors.settings.macros.name",
|
|
||||||
hint: "smart-doors.settings.macros.hint",
|
|
||||||
scope: "world",
|
|
||||||
config: true,
|
|
||||||
type: Boolean,
|
|
||||||
default: true,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user