Compare commits
128 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d79a808c81 | ||
|
|
1016c39bdf | ||
|
|
5f4923e049 | ||
|
|
e83237a728 | ||
|
|
9f323738bf | ||
|
|
1fc76fa50d | ||
|
|
36d5ff0a88 | ||
|
|
d05874ed13 | ||
|
|
df2521e69c | ||
|
|
3c6a685fe8 | ||
|
|
a77e560284 | ||
|
|
4cec83918f | ||
|
|
fec95c51c6 | ||
|
|
425c42d049 | ||
|
|
ab6f0c4f5b | ||
|
|
5d6e57b896 | ||
|
|
7c0a8125f2 | ||
|
|
7481ef08a8 | ||
|
|
b578dd5fb0 | ||
|
|
5d6f934d88 | ||
|
|
337f0bfa8a | ||
|
|
c62784894b | ||
|
|
75fff43d7d | ||
|
|
a9eeeac0df | ||
|
|
c8af0ff0a9 | ||
|
|
9e200db7b9 | ||
|
|
c08cf83096 | ||
|
|
d9368b06d0 | ||
|
|
2703367681 | ||
|
|
d419442549 | ||
|
|
99df01c950 | ||
|
|
d76349b3bb | ||
|
|
39c061f4e8 | ||
|
|
6d167ddb22 | ||
|
|
037acbd459 | ||
|
|
4d3fc3bb09 | ||
|
|
4b984d4fac | ||
|
|
58843613ba | ||
|
|
39b549b24b | ||
|
|
c79177de72 | ||
|
|
11d09b1487 | ||
|
|
0e4918d57d | ||
|
|
949f313af2 | ||
|
|
85b63f152f | ||
|
|
2141d52a7a | ||
|
|
4c84235064 | ||
|
|
0e194a5408 | ||
|
|
4b60eac330 | ||
|
|
cb017c359d | ||
|
|
15aaaa5c14 | ||
|
|
290bee83b4 | ||
|
|
fcd2461205 | ||
|
|
52198d0249 | ||
|
|
ba7ccfdfa0 | ||
|
|
92d3b086fa | ||
|
|
e180595dcd | ||
|
|
ed708bdde0 | ||
|
|
d7852d640f | ||
|
|
a2fdee88b6 | ||
|
|
af070b1578 | ||
|
|
83a8eeef0f | ||
|
|
34f8e7402b | ||
|
|
1b7e2cd850 | ||
|
|
463b7f0fc9 | ||
|
|
266495abc8 | ||
|
|
453d4365d3 | ||
|
|
e0ce6275bf | ||
|
|
16b16ce6c6 | ||
|
|
80c72a274e | ||
|
|
91f0f7954c | ||
|
|
c74abcb608 | ||
|
|
da8b91594e | ||
|
|
fc26f5a73e | ||
|
|
4f60766d5d | ||
|
|
e992aeebef | ||
|
|
4108346a98 | ||
|
|
946fadadc2 | ||
|
|
d2cc2833a9 | ||
|
|
af57326194 | ||
|
|
98c69e9e17 | ||
|
|
395edd0563 | ||
|
|
43e87e7786 | ||
|
|
ad347504c6 | ||
|
|
4e6e99b695 | ||
|
|
104624a322 | ||
|
|
79d166e6af | ||
|
|
86c934e8ac | ||
|
|
a034cbf30e | ||
|
|
d5680ebf8a | ||
|
|
53f2fcc945 | ||
|
|
612e127be4 | ||
|
|
2b0d975cee | ||
|
|
248ab9bb6b | ||
|
|
314ce85410 | ||
|
|
9ff45dbcc2 | ||
|
|
6caf19bc99 | ||
|
|
c2b04d0977 | ||
|
|
9012c4a558 | ||
|
|
65a84937f2 | ||
|
|
eebb88b6b1 | ||
|
|
06ab7c5116 | ||
|
|
89f03c7601 | ||
|
|
9d2eb14c0c | ||
|
|
7b3cb54983 | ||
|
|
a09bad2fed | ||
|
|
afd897edfe | ||
|
|
efc79cb6e7 | ||
|
|
35efe39ea7 | ||
|
|
034067bd6e | ||
|
|
0d75cd5d15 | ||
|
|
4f1376a666 | ||
|
|
78b1d71b9d | ||
|
|
1323d8006c | ||
|
|
87d722adaf | ||
|
|
90e511eb00 | ||
|
|
5b8c25f5de | ||
|
|
2fbc54fee8 | ||
|
|
a064ae3fe8 | ||
|
|
ba9b518d7e | ||
|
|
2f729070b2 | ||
|
|
7aedb9451c | ||
|
|
c6886dd49e | ||
|
|
038ce490e4 | ||
|
|
52bef57637 | ||
|
|
29e9f8c8dc | ||
|
|
fea02811ff | ||
|
|
73cee52fff | ||
|
|
b58c006ed4 |
60
README.md
60
README.md
@@ -1,58 +1,4 @@
|
|||||||
TODO
|
RPG Docs
|
||||||
====
|
========
|
||||||
|
|
||||||
* Get Polymer installed using bower.
|
This is the repo for [DiceCloud](dicecloud.com). The currently deployed version should always be the head of the master branch.
|
||||||
* Install Vulcanize package listed below
|
|
||||||
* Copy the differential polymer demo to get polymer implemented nicely
|
|
||||||
* Update Meteor
|
|
||||||
* Install and use LESS
|
|
||||||
|
|
||||||
Packages used
|
|
||||||
=============
|
|
||||||
|
|
||||||
* meteor-platform
|
|
||||||
* Base Meteor.
|
|
||||||
* [Docs](http://docs.meteor.com/#/full/)
|
|
||||||
* autopublish
|
|
||||||
* Publishes everything to the client.
|
|
||||||
* Must be removed before release
|
|
||||||
* insecure
|
|
||||||
* Allows the client the freedom to modify any colleciton.
|
|
||||||
* Must be removed before release
|
|
||||||
* iron:router
|
|
||||||
* Enables pagination and URL's to direct to specific templates.
|
|
||||||
* [Tutorial](http://www.manuel-schoebel.com/blog/iron-router-tutorial)
|
|
||||||
* accounts-password
|
|
||||||
* Lets users create accounts with a simple password
|
|
||||||
* accounts-ui
|
|
||||||
* Adds simple UI for logging in
|
|
||||||
* random
|
|
||||||
* Somewhat decent cryptographically strong psuedo random number generation.
|
|
||||||
* [readme](https://atmospherejs.com/meteor/random)
|
|
||||||
* dburles:collection-helpers
|
|
||||||
* Adds template-style helpers to collections. [github page](https://github.com/dburles/meteor-collection-helpers)
|
|
||||||
* reactive-var
|
|
||||||
* Friendly reactive variables
|
|
||||||
* [Meteor Docs](http://docs.meteor.com/#/full/reactivevar_pkg)
|
|
||||||
* cw4gn3r:jquery-event-drag
|
|
||||||
* Adds jquery drag events
|
|
||||||
* underscore
|
|
||||||
* Handy javascript utilities
|
|
||||||
* [Docs](http://underscorejs.org/)
|
|
||||||
* aldeed:collection2
|
|
||||||
* Extends collections with Schemas
|
|
||||||
* [(gitHub page)](https://github.com/aldeed/meteor-collection2)
|
|
||||||
* uses [SimpleSchema](https://github.com/aldeed/meteor-simple-schema)
|
|
||||||
* aldeed:autoform
|
|
||||||
* Automatically generates bootstrap forms for collection2 Schemas.
|
|
||||||
* [github](https://github.com/aldeed/meteor-autoform)
|
|
||||||
* differential:vulcanize
|
|
||||||
* Bakes all the polymer imports into one file
|
|
||||||
* [github](https://github.com/Differential/meteor-vulcanize)
|
|
||||||
|
|
||||||
************
|
|
||||||
|
|
||||||
Resources
|
|
||||||
=========
|
|
||||||
|
|
||||||
[differential's polymer demo](https://github.com/Differential/polymer-demo)
|
|
||||||
|
|||||||
4
rpg-docs/.gitignore
vendored
4
rpg-docs/.gitignore
vendored
@@ -1,5 +1,7 @@
|
|||||||
.meteor/local
|
.meteor/local
|
||||||
.meteor/meteorite
|
.meteor/meteorite
|
||||||
|
.demeteorized
|
||||||
|
settings.json
|
||||||
public/components
|
public/components
|
||||||
nohup.out
|
nohup.out
|
||||||
dump
|
dump
|
||||||
|
|||||||
@@ -6,3 +6,7 @@ notices-for-0.9.0
|
|||||||
notices-for-0.9.1
|
notices-for-0.9.1
|
||||||
0.9.4-platform-file
|
0.9.4-platform-file
|
||||||
notices-for-facebook-graph-api-2
|
notices-for-facebook-graph-api-2
|
||||||
|
1.2.0-standard-minifiers-package
|
||||||
|
1.2.0-meteor-platform-split
|
||||||
|
1.2.0-cordova-changes
|
||||||
|
1.2.0-breaking-changes
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
|
|
||||||
|
|||||||
@@ -3,7 +3,6 @@
|
|||||||
# 'meteor add' and 'meteor remove' will edit this file for you,
|
# 'meteor add' and 'meteor remove' will edit this file for you,
|
||||||
# but you can also edit it by hand.
|
# but you can also edit it by hand.
|
||||||
|
|
||||||
meteor-platform
|
|
||||||
iron:router
|
iron:router
|
||||||
accounts-password
|
accounts-password
|
||||||
accounts-ui
|
accounts-ui
|
||||||
@@ -12,7 +11,6 @@ dburles:collection-helpers
|
|||||||
reactive-var
|
reactive-var
|
||||||
underscore
|
underscore
|
||||||
aldeed:collection2
|
aldeed:collection2
|
||||||
differential:vulcanize
|
|
||||||
matb33:collection-hooks
|
matb33:collection-hooks
|
||||||
zimme:collection-softremovable
|
zimme:collection-softremovable
|
||||||
momentjs:moment
|
momentjs:moment
|
||||||
@@ -25,3 +23,22 @@ splendido:accounts-meld
|
|||||||
email
|
email
|
||||||
fourseven:scss@2.1.1
|
fourseven:scss@2.1.1
|
||||||
wolves:bourbon
|
wolves:bourbon
|
||||||
|
meteorhacks:subs-manager
|
||||||
|
meteorhacks:kadira
|
||||||
|
chuangbo:marked
|
||||||
|
reywood:iron-router-ga
|
||||||
|
standard-minifiers
|
||||||
|
meteor-base
|
||||||
|
mobile-experience
|
||||||
|
mongo
|
||||||
|
blaze-html-templates
|
||||||
|
session
|
||||||
|
jquery
|
||||||
|
tracker
|
||||||
|
logging
|
||||||
|
reload
|
||||||
|
ejson
|
||||||
|
spacebars
|
||||||
|
check
|
||||||
|
useraccounts:iron-routing
|
||||||
|
wizonesolutions:canonical
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
METEOR@1.1.0.2
|
METEOR@1.2.0.2
|
||||||
|
|||||||
@@ -1,90 +1,116 @@
|
|||||||
accounts-base@1.2.0
|
accounts-base@1.2.1
|
||||||
accounts-google@1.0.4
|
accounts-google@1.0.6
|
||||||
accounts-oauth@1.1.5
|
accounts-oauth@1.1.7
|
||||||
accounts-password@1.1.1
|
accounts-password@1.1.3
|
||||||
accounts-ui@1.1.5
|
accounts-ui@1.1.6
|
||||||
accounts-ui-unstyled@1.1.7
|
accounts-ui-unstyled@1.1.8
|
||||||
aldeed:collection2@2.3.3
|
aldeed:collection2@2.5.0
|
||||||
aldeed:simple-schema@1.3.3
|
aldeed:simple-schema@1.3.3
|
||||||
autoupdate@1.2.1
|
autoupdate@1.2.3
|
||||||
base64@1.0.3
|
babel-compiler@5.8.24_1
|
||||||
binary-heap@1.0.3
|
babel-runtime@0.1.4
|
||||||
blaze@2.1.2
|
base64@1.0.4
|
||||||
blaze-tools@1.0.3
|
binary-heap@1.0.4
|
||||||
boilerplate-generator@1.0.3
|
blaze@2.1.3
|
||||||
callback-hook@1.0.3
|
blaze-html-templates@1.0.1
|
||||||
check@1.0.5
|
blaze-tools@1.0.4
|
||||||
coffeescript@1.0.6
|
boilerplate-generator@1.0.4
|
||||||
|
caching-compiler@1.0.0
|
||||||
|
caching-html-compiler@1.0.2
|
||||||
|
callback-hook@1.0.4
|
||||||
|
check@1.0.6
|
||||||
|
chuangbo:marked@0.3.5_1
|
||||||
|
coffeescript@1.0.10
|
||||||
dburles:collection-helpers@1.0.3
|
dburles:collection-helpers@1.0.3
|
||||||
dburles:mongo-collection-instances@0.3.3
|
dburles:mongo-collection-instances@0.3.4
|
||||||
ddp@1.1.0
|
ddp@1.2.2
|
||||||
deps@1.0.7
|
ddp-client@1.2.1
|
||||||
differential:vulcanize@0.0.5
|
ddp-common@1.2.1
|
||||||
|
ddp-rate-limiter@1.0.0
|
||||||
|
ddp-server@1.2.1
|
||||||
|
deps@1.0.9
|
||||||
|
diff-sequence@1.0.1
|
||||||
|
ecmascript@0.1.5
|
||||||
|
ecmascript-collections@0.1.6
|
||||||
ecwyne:mathjs@0.25.0
|
ecwyne:mathjs@0.25.0
|
||||||
ejson@1.0.6
|
ejson@1.0.7
|
||||||
email@1.0.6
|
email@1.0.7
|
||||||
fastclick@1.0.3
|
fastclick@1.0.7
|
||||||
fourseven:scss@2.1.1
|
fourseven:scss@2.1.1
|
||||||
geojson-utils@1.0.3
|
geojson-utils@1.0.4
|
||||||
google@1.1.5
|
google@1.1.6
|
||||||
html-tools@1.0.4
|
hot-code-push@1.0.0
|
||||||
htmljs@1.0.4
|
html-tools@1.0.5
|
||||||
http@1.1.0
|
htmljs@1.0.5
|
||||||
id-map@1.0.3
|
http@1.1.1
|
||||||
iron:controller@1.0.7
|
id-map@1.0.4
|
||||||
iron:core@1.0.7
|
iron:controller@1.0.8
|
||||||
iron:dynamic-template@1.0.7
|
iron:core@1.0.8
|
||||||
iron:layout@1.0.7
|
iron:dynamic-template@1.0.8
|
||||||
iron:location@1.0.7
|
iron:layout@1.0.8
|
||||||
iron:middleware-stack@1.0.7
|
iron:location@1.0.9
|
||||||
iron:router@1.0.7
|
iron:middleware-stack@1.0.9
|
||||||
iron:url@1.0.7
|
iron:router@1.0.9
|
||||||
jquery@1.11.3_2
|
iron:url@1.0.9
|
||||||
json@1.0.3
|
jquery@1.11.4
|
||||||
lai:collection-extensions@0.1.3
|
lai:collection-extensions@0.1.4
|
||||||
launch-screen@1.0.2
|
launch-screen@1.0.4
|
||||||
less@1.0.14
|
less@2.5.0_3
|
||||||
livedata@1.0.13
|
livedata@1.0.15
|
||||||
localstorage@1.0.3
|
localstorage@1.0.5
|
||||||
logging@1.0.7
|
logging@1.0.8
|
||||||
matb33:collection-hooks@0.7.13
|
matb33:collection-hooks@0.8.1
|
||||||
meteor@1.1.6
|
meteor@1.1.9
|
||||||
meteor-platform@1.2.2
|
meteor-base@1.0.1
|
||||||
minifiers@1.1.5
|
meteorhacks:kadira@2.23.4
|
||||||
minimongo@1.0.8
|
meteorhacks:meteorx@1.3.1
|
||||||
mobile-status-bar@1.0.3
|
meteorhacks:subs-manager@1.6.2
|
||||||
momentjs:moment@2.10.3
|
minifiers@1.1.7
|
||||||
mongo@1.1.0
|
minimongo@1.0.10
|
||||||
|
mobile-experience@1.0.1
|
||||||
|
mobile-status-bar@1.0.6
|
||||||
|
momentjs:moment@2.10.6
|
||||||
|
mongo@1.1.2
|
||||||
|
mongo-id@1.0.1
|
||||||
|
mongo-livedata@1.0.9
|
||||||
npm-bcrypt@0.7.8_2
|
npm-bcrypt@0.7.8_2
|
||||||
oauth@1.1.4
|
npm-mongo@1.4.39_1
|
||||||
oauth2@1.1.3
|
oauth@1.1.6
|
||||||
observe-sequence@1.0.6
|
oauth2@1.1.5
|
||||||
ordered-dict@1.0.3
|
observe-sequence@1.0.7
|
||||||
percolate:migrations@0.7.5
|
ordered-dict@1.0.4
|
||||||
random@1.0.3
|
percolate:migrations@0.9.6
|
||||||
reactive-dict@1.1.0
|
promise@0.5.0
|
||||||
reactive-var@1.0.5
|
random@1.0.4
|
||||||
reload@1.1.3
|
rate-limit@1.0.0
|
||||||
retry@1.0.3
|
reactive-dict@1.1.2
|
||||||
routepolicy@1.0.5
|
reactive-var@1.0.6
|
||||||
service-configuration@1.0.4
|
reload@1.1.4
|
||||||
session@1.1.0
|
retry@1.0.4
|
||||||
sha@1.0.3
|
reywood:iron-router-ga@0.7.1
|
||||||
|
routepolicy@1.0.6
|
||||||
|
service-configuration@1.0.5
|
||||||
|
session@1.1.1
|
||||||
|
sha@1.0.4
|
||||||
softwarerero:accounts-t9n@1.0.9
|
softwarerero:accounts-t9n@1.0.9
|
||||||
spacebars@1.0.6
|
spacebars@1.0.7
|
||||||
spacebars-compiler@1.0.6
|
spacebars-compiler@1.0.7
|
||||||
splendido:accounts-emails-field@1.2.0
|
splendido:accounts-emails-field@1.2.0
|
||||||
splendido:accounts-meld@1.3.0
|
splendido:accounts-meld@1.3.1
|
||||||
srp@1.0.3
|
srp@1.0.4
|
||||||
templating@1.1.1
|
standard-minifiers@1.0.1
|
||||||
tracker@1.0.7
|
templating@1.1.4
|
||||||
ui@1.0.6
|
templating-tools@1.0.0
|
||||||
underscore@1.0.3
|
tracker@1.0.9
|
||||||
url@1.0.4
|
ui@1.0.8
|
||||||
useraccounts:core@1.9.1
|
underscore@1.0.4
|
||||||
useraccounts:polymer@1.9.1
|
url@1.0.5
|
||||||
webapp@1.2.0
|
useraccounts:core@1.12.3
|
||||||
webapp-hashing@1.0.3
|
useraccounts:iron-routing@1.12.3
|
||||||
wolves:bourbon@1.0.0
|
useraccounts:polymer@1.12.3
|
||||||
zimme:collection-behaviours@1.0.4
|
webapp@1.2.2
|
||||||
|
webapp-hashing@1.0.5
|
||||||
|
wizonesolutions:canonical@0.0.5
|
||||||
|
wolves:bourbon@1.2.0
|
||||||
|
zimme:collection-behaviours@1.1.3
|
||||||
zimme:collection-softremovable@1.0.4
|
zimme:collection-softremovable@1.0.4
|
||||||
|
|||||||
@@ -69,5 +69,17 @@ Attacks.attachSchema(Schemas.Attack);
|
|||||||
Attacks.attachBehaviour("softRemovable");
|
Attacks.attachBehaviour("softRemovable");
|
||||||
makeChild(Attacks, ["name", "enabled"]); //children of lots of things
|
makeChild(Attacks, ["name", "enabled"]); //children of lots of things
|
||||||
|
|
||||||
|
Attacks.after.insert(function (userId, attack) {
|
||||||
|
//Check to see if this attack's parent is a spell, if so, mirror prepared state to enabled
|
||||||
|
if (attack.parent.collection === "Spells") {
|
||||||
|
var parentSpell = Spells.findOne(attack.parent.id);
|
||||||
|
if (parentSpell.prepared === "unprepared") {
|
||||||
|
Attacks.update(attack._id, {$set: {enabled: false}});
|
||||||
|
} else if (parentSpell.prepared === "prepared" || "always") {
|
||||||
|
Attacks.update(attack._id, {$set: {enabled: true}});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
Attacks.allow(CHARACTER_SUBSCHEMA_ALLOW);
|
Attacks.allow(CHARACTER_SUBSCHEMA_ALLOW);
|
||||||
Attacks.deny(CHARACTER_SUBSCHEMA_DENY);
|
Attacks.deny(CHARACTER_SUBSCHEMA_DENY);
|
||||||
|
|||||||
@@ -1,28 +1,51 @@
|
|||||||
Buffs = new Mongo.Collection("buffs");
|
Buffs = new Mongo.Collection("buffs");
|
||||||
|
|
||||||
//buffs are temporary once applied and store things which expire and their expiry time
|
|
||||||
Schemas.Buff = new SimpleSchema({
|
Schemas.Buff = new SimpleSchema({
|
||||||
//buff id
|
|
||||||
_id: {
|
|
||||||
type: String,
|
|
||||||
regEx: SimpleSchema.RegEx.Id,
|
|
||||||
autoValue: function(){
|
|
||||||
if (!this.isSet) return Random.id();
|
|
||||||
},
|
|
||||||
},
|
|
||||||
charId: {
|
charId: {
|
||||||
type: String,
|
type: String,
|
||||||
regEx: SimpleSchema.RegEx.Id,
|
regEx: SimpleSchema.RegEx.Id,
|
||||||
},
|
},
|
||||||
//expiry time
|
name: {
|
||||||
expiry: {type: Number, optional: true},
|
type: String,
|
||||||
duration: {type: Number},
|
trim: false,
|
||||||
|
},
|
||||||
|
description: {
|
||||||
|
type: String,
|
||||||
|
optional: true,
|
||||||
|
trim: false,
|
||||||
|
},
|
||||||
|
enabled: {
|
||||||
|
type: Boolean,
|
||||||
|
defaultValue: true,
|
||||||
|
},
|
||||||
|
type: {
|
||||||
|
type: String,
|
||||||
|
allowedValues: [
|
||||||
|
"inate",
|
||||||
|
"custom",
|
||||||
|
],
|
||||||
|
},
|
||||||
|
"lifeTime.total": {
|
||||||
|
type: Number,
|
||||||
|
defaultValue: 0, //0 is infinite
|
||||||
|
min: 0,
|
||||||
|
},
|
||||||
|
"lifeTime.spent": {
|
||||||
|
type: Number,
|
||||||
|
defaultValue: 0,
|
||||||
|
min: 0,
|
||||||
|
},
|
||||||
|
color: {
|
||||||
|
type: String,
|
||||||
|
allowedValues: _.pluck(colorOptions, "key"),
|
||||||
|
defaultValue: "q",
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
Buffs.attachSchema(Schemas.Buff);
|
Buffs.attachSchema(Schemas.Buff);
|
||||||
|
|
||||||
Buffs.attachBehaviour("softRemovable");
|
Buffs.attachBehaviour("softRemovable");
|
||||||
makeParent(Buffs, "name"); //parents of effects and attacks
|
makeParent(Buffs, ["name", "enabled"]); //parents of effects
|
||||||
|
|
||||||
Buffs.allow(CHARACTER_SUBSCHEMA_ALLOW);
|
Buffs.allow(CHARACTER_SUBSCHEMA_ALLOW);
|
||||||
Buffs.deny(CHARACTER_SUBSCHEMA_DENY);
|
Buffs.deny(CHARACTER_SUBSCHEMA_DENY);
|
||||||
|
|||||||
@@ -3,16 +3,17 @@ Characters = new Mongo.Collection("characters");
|
|||||||
|
|
||||||
Schemas.Character = new SimpleSchema({
|
Schemas.Character = new SimpleSchema({
|
||||||
//strings
|
//strings
|
||||||
name: {type: String, defaultValue: "", trim: false},
|
name: {type: String, defaultValue: "", trim: false, optional: true},
|
||||||
alignment: {type: String, defaultValue: "", trim: false},
|
alignment: {type: String, defaultValue: "", trim: false, optional: true},
|
||||||
gender: {type: String, defaultValue: "", trim: false},
|
gender: {type: String, defaultValue: "", trim: false, optional: true},
|
||||||
race: {type: String, defaultValue: "", trim: false},
|
race: {type: String, defaultValue: "", trim: false, optional: true},
|
||||||
description: {type: String, defaultValue: "", trim: false},
|
picture: {type: String, defaultValue: "", trim: true, optional: true},
|
||||||
personality: {type: String, defaultValue: "", trim: false},
|
description: {type: String, defaultValue: "", trim: false, optional: true},
|
||||||
ideals: {type: String, defaultValue: "", trim: false},
|
personality: {type: String, defaultValue: "", trim: false, optional: true},
|
||||||
bonds: {type: String, defaultValue: "", trim: false},
|
ideals: {type: String, defaultValue: "", trim: false, optional: true},
|
||||||
flaws: {type: String, defaultValue: "", trim: false},
|
bonds: {type: String, defaultValue: "", trim: false, optional: true},
|
||||||
backstory: {type: String, defaultValue: "", trim: false},
|
flaws: {type: String, defaultValue: "", trim: false, optional: true},
|
||||||
|
backstory: {type: String, defaultValue: "", trim: false, optional: true},
|
||||||
|
|
||||||
//attributes
|
//attributes
|
||||||
//ability scores
|
//ability scores
|
||||||
@@ -32,6 +33,7 @@ Schemas.Character = new SimpleSchema({
|
|||||||
age: {type: Schemas.Attribute},
|
age: {type: Schemas.Attribute},
|
||||||
ageRate: {type: Schemas.Attribute},
|
ageRate: {type: Schemas.Attribute},
|
||||||
armor: {type: Schemas.Attribute},
|
armor: {type: Schemas.Attribute},
|
||||||
|
carryMultiplier: {type: Schemas.Attribute},
|
||||||
|
|
||||||
//resources
|
//resources
|
||||||
level1SpellSlots: {type: Schemas.Attribute},
|
level1SpellSlots: {type: Schemas.Attribute},
|
||||||
@@ -159,6 +161,7 @@ Schemas.Character = new SimpleSchema({
|
|||||||
deathSave: {type: Schemas.DeathSave},
|
deathSave: {type: Schemas.DeathSave},
|
||||||
|
|
||||||
//permissions
|
//permissions
|
||||||
|
party: {type: String, regEx: SimpleSchema.RegEx.Id, optional: true},
|
||||||
owner: {type: String, regEx: SimpleSchema.RegEx.Id},
|
owner: {type: String, regEx: SimpleSchema.RegEx.Id},
|
||||||
readers: {type: [String], regEx: SimpleSchema.RegEx.Id, defaultValue: []},
|
readers: {type: [String], regEx: SimpleSchema.RegEx.Id, defaultValue: []},
|
||||||
writers: {type: [String], regEx: SimpleSchema.RegEx.Id, defaultValue: []},
|
writers: {type: [String], regEx: SimpleSchema.RegEx.Id, defaultValue: []},
|
||||||
@@ -168,61 +171,122 @@ Schemas.Character = new SimpleSchema({
|
|||||||
defaultValue: "q",
|
defaultValue: "q",
|
||||||
},
|
},
|
||||||
//TODO add per-character settings
|
//TODO add per-character settings
|
||||||
"settings.experiencesInc": {type: Number, defaultValue: 20}, //how many experiences to load at a time in XP table
|
//how many experiences to load at a time in XP table
|
||||||
|
"settings.experiencesInc": {type: Number, defaultValue: 20},
|
||||||
|
//slowed down by carrying too much?
|
||||||
|
"settings.useVariantEncumbrance": {type: Boolean, defaultValue: false},
|
||||||
|
"settings.useStandardEncumbrance": {type: Boolean, defaultValue: true},
|
||||||
|
//hide spellcasting
|
||||||
|
"settings.hideSpellcasting": {type: Boolean, defaultValue: false},
|
||||||
|
//show to anyone with link
|
||||||
|
"settings.viewPermission": {
|
||||||
|
type: String,
|
||||||
|
defaultValue: "whitelist",
|
||||||
|
allowedValues: ["whitelist", "public"],
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
Characters.attachSchema(Schemas.Character);
|
Characters.attachSchema(Schemas.Character);
|
||||||
|
|
||||||
var attributeBase = function(charId, statName){
|
var attributeBase = preventLoop(function(charId, statName){
|
||||||
check(statName, String);
|
check(statName, String);
|
||||||
var effects = Effects.find(
|
//if it's a damage multiplier, we treat it specially
|
||||||
{charId: charId, stat: statName, enabled: true}
|
if (_.contains(DAMAGE_MULTIPLIERS, statName)){
|
||||||
).fetch();
|
var invulnerabilityCount = Effects.find({
|
||||||
effects = _.groupBy(effects, "operation");
|
charId: charId,
|
||||||
var value = _.contains(DAMAGE_MULTIPLIERS, statName) ? 1 : 0;
|
stat: statName,
|
||||||
|
enabled: true,
|
||||||
|
operation: "mul",
|
||||||
|
value: 0,
|
||||||
|
}).count();
|
||||||
|
if (invulnerabilityCount) return 0;
|
||||||
|
var resistCount = Effects.find({
|
||||||
|
charId: charId,
|
||||||
|
stat: statName,
|
||||||
|
enabled: true,
|
||||||
|
operation: "mul",
|
||||||
|
value: 0.5,
|
||||||
|
}).count();
|
||||||
|
var vulnCount = Effects.find({
|
||||||
|
charId: charId,
|
||||||
|
stat: statName,
|
||||||
|
enabled: true,
|
||||||
|
operation: "mul",
|
||||||
|
value: 2,
|
||||||
|
}).count();
|
||||||
|
if (!resistCount && !vulnCount){
|
||||||
|
return 1;
|
||||||
|
} else if (resistCount && !vulnCount){
|
||||||
|
return 0.5;
|
||||||
|
} else if (!resistCount && vulnCount){
|
||||||
|
return 2;
|
||||||
|
} else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var value;
|
||||||
|
var base = 0;
|
||||||
|
var add = 0;
|
||||||
|
var mul = 1;
|
||||||
|
var min = Number.NEGATIVE_INFINITY;
|
||||||
|
var max = Number.POSITIVE_INFINITY;
|
||||||
|
|
||||||
//start with the highest base value
|
Effects.find({
|
||||||
_.each(effects.base, function(effect){
|
charId: charId,
|
||||||
var efv = evaluateEffect(charId, effect);
|
stat: statName,
|
||||||
if (efv > value){
|
enabled: true,
|
||||||
value = efv;
|
operation: {$in: ["base", "add", "mul", "min", "max"]},
|
||||||
|
}).forEach(function(effect) {
|
||||||
|
value = evaluateEffect(charId, effect);
|
||||||
|
if (effect.operation === "base"){
|
||||||
|
if (value > base) base = value;
|
||||||
|
} else if (effect.operation === "add"){
|
||||||
|
add += value;
|
||||||
|
} else if (effect.operation === "mul"){
|
||||||
|
mul *= value;
|
||||||
|
} else if (effect.operation === "min"){
|
||||||
|
if (value > min) min = value;
|
||||||
|
} else if (effect.operation === "max"){
|
||||||
|
if (value < max) max = value;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
//add all the add values
|
var result = (base + add) * mul;
|
||||||
_.each(effects.add, function(effect){
|
if (result < min) result = min;
|
||||||
value += evaluateEffect(charId, effect);
|
if (result > max) result = max;
|
||||||
});
|
|
||||||
|
|
||||||
//multiply all the mul values
|
return Math.floor(result);
|
||||||
_.each(effects.mul, function(effect){
|
});
|
||||||
value *= evaluateEffect(charId, effect);
|
|
||||||
});
|
|
||||||
|
|
||||||
//ensure value is >= all mins
|
if (Meteor.isClient) {
|
||||||
_.each(effects.min, function(effect){
|
Template.registerHelper("characterCalculate", function(func, charId, input) {
|
||||||
var min = evaluateEffect(charId, effect);
|
try {
|
||||||
value = value > min ? value : min;
|
return Characters.calculate[func](charId, input);
|
||||||
|
} catch (e){
|
||||||
|
if (!Characters.calculate[func]){
|
||||||
|
throw new Error(func + "is not a function name");
|
||||||
|
} else {
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
//ensure value is <= all maxes
|
//create a local memoize with a argument concatenating hash function
|
||||||
_.each(effects.max, function(effect){
|
var memoize = function(f) {
|
||||||
var max = evaluateEffect(charId, effect);
|
return Tracker.memoize(f, function() {
|
||||||
value = value < max ? value : max;
|
return _.reduce(arguments, function(memo, arg) {
|
||||||
|
return memo + arg;
|
||||||
|
}, "");
|
||||||
});
|
});
|
||||||
return value;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
//functions and calculated values.
|
//memoize funcitons that have finds and slow loops
|
||||||
//These functions can only rely on this._id since no other
|
Characters.calculate = {
|
||||||
//field is likely to be attached to all returned characters
|
getField: function(charId, fieldName) {
|
||||||
Characters.helpers({
|
|
||||||
//returns the value stored in the field requested
|
|
||||||
//will set up dependencies on just that field
|
|
||||||
getField : function(fieldName){
|
|
||||||
var fieldSelector = {};
|
var fieldSelector = {};
|
||||||
fieldSelector[fieldName] = 1;
|
fieldSelector[fieldName] = 1;
|
||||||
var char = Characters.findOne(this._id, {fields: fieldSelector});
|
var char = Characters.findOne(charId, {fields: fieldSelector});
|
||||||
var field = char[fieldName];
|
var field = char[fieldName];
|
||||||
if (field === undefined){
|
if (field === undefined){
|
||||||
throw new Meteor.Error(
|
throw new Meteor.Error(
|
||||||
@@ -235,8 +299,7 @@ Characters.helpers({
|
|||||||
}
|
}
|
||||||
return field;
|
return field;
|
||||||
},
|
},
|
||||||
//returns the value of a field
|
fieldValue: function(charId, fieldName) {
|
||||||
fieldValue : function(fieldName){
|
|
||||||
if (!Schemas.Character.schema(fieldName)){
|
if (!Schemas.Character.schema(fieldName)){
|
||||||
throw new Meteor.Error(
|
throw new Meteor.Error(
|
||||||
"Field not found",
|
"Field not found",
|
||||||
@@ -246,102 +309,92 @@ Characters.helpers({
|
|||||||
//duck typing to get the right value function
|
//duck typing to get the right value function
|
||||||
//.ability implies skill
|
//.ability implies skill
|
||||||
if (Schemas.Character.schema(fieldName + ".ability")){
|
if (Schemas.Character.schema(fieldName + ".ability")){
|
||||||
return this.skillMod(fieldName);
|
return Characters.calculate.skillMod(charId, fieldName);
|
||||||
}
|
}
|
||||||
//adjustment implies attribute
|
//adjustment implies attribute
|
||||||
if (Schemas.Character.schema(fieldName + ".adjustment")){
|
if (Schemas.Character.schema(fieldName + ".adjustment")){
|
||||||
return this.attributeValue(fieldName);
|
return Characters.calculate.attributeValue(charId, fieldName);
|
||||||
}
|
}
|
||||||
//fall back to just returning the field itself
|
//fall back to just returning the field itself
|
||||||
return this.getField(fieldName);
|
return Characters.calculate.getField(charId, fieldName);
|
||||||
},
|
},
|
||||||
|
attributeValue: memoize(function(charId, attributeName){
|
||||||
attributeValue: function(attributeName){
|
var attribute = Characters.calculate.getField(charId, attributeName);
|
||||||
var charId = this._id;
|
|
||||||
var attribute = this.getField(attributeName);
|
|
||||||
//base value
|
//base value
|
||||||
var value = this.attributeBase(attributeName);
|
var value = Characters.calculate.attributeBase(charId, attributeName);
|
||||||
//plus adjustment
|
//plus adjustment
|
||||||
value += attribute.adjustment;
|
value += attribute.adjustment;
|
||||||
return value;
|
return value;
|
||||||
},
|
}),
|
||||||
|
attributeBase: memoize(function(charId, attributeName){
|
||||||
attributeBase: preventLoop(function(attributeName){
|
|
||||||
var charId = this._id;
|
|
||||||
//base value
|
|
||||||
return attributeBase(charId, attributeName);
|
return attributeBase(charId, attributeName);
|
||||||
}),
|
}),
|
||||||
|
skillMod: memoize(preventLoop(function(charId, skillName){
|
||||||
skillMod: preventLoop(function(skillName){
|
var skill = Characters.calculate.getField(charId, skillName);
|
||||||
var charId = this._id;
|
|
||||||
var skill = this.getField(skillName);
|
|
||||||
//get the final value of the ability score
|
//get the final value of the ability score
|
||||||
var ability = this.attributeValue(skill.ability);
|
var ability = Characters.calculate.attributeValue(charId, skill.ability);
|
||||||
|
|
||||||
//base modifier
|
//base modifier
|
||||||
var mod = +getMod(ability);
|
var mod = +getMod(ability);
|
||||||
|
|
||||||
//multiply proficiency bonus by largest value in proficiency array
|
//multiply proficiency bonus by largest value in proficiency array
|
||||||
var prof = this.proficiency(skillName);
|
var prof = Characters.calculate.proficiency(charId, skillName);
|
||||||
|
|
||||||
//add multiplied proficiency bonus to modifier
|
//add multiplied proficiency bonus to modifier
|
||||||
mod += prof * this.attributeValue("proficiencyBonus");
|
mod += prof * Characters.calculate.attributeValue(charId, "proficiencyBonus");
|
||||||
|
|
||||||
//apply all effects
|
//apply all effects
|
||||||
var rawEffects = Effects.find(
|
var value;
|
||||||
{charId: charId, stat: skillName, enabled: true}
|
var add = 0;
|
||||||
).fetch();
|
var mul = 1;
|
||||||
var effects = _.groupBy(rawEffects, "operation");
|
var min = Number.NEGATIVE_INFINITY;
|
||||||
_.forEach(effects.add, function(effect){
|
var max = Number.POSITIVE_INFINITY;
|
||||||
mod += evaluateEffect(charId, effect);
|
|
||||||
});
|
|
||||||
_.forEach(effects.mul, function(effect){
|
|
||||||
mod *= evaluateEffect(charId, effect);
|
|
||||||
});
|
|
||||||
_.forEach(effects.min, function(effect){
|
|
||||||
var min = evaluateEffect(charId, effect);
|
|
||||||
mod = mod > min ? mod : min;
|
|
||||||
});
|
|
||||||
_.forEach(effects.max, function(effect){
|
|
||||||
var max = evaluateEffect(charId, effect);
|
|
||||||
mod = mod < max ? mod : max;
|
|
||||||
});
|
|
||||||
return signedString(mod);
|
|
||||||
}),
|
|
||||||
|
|
||||||
proficiency: function(skillName){
|
Effects.find({
|
||||||
var charId = this._id;
|
charId: charId,
|
||||||
//return largest value in proficiency array
|
stat: skillName,
|
||||||
var prof = 0;
|
enabled: true,
|
||||||
Proficiencies.find(
|
operation: {$in: ["base", "add", "mul", "min", "max"]},
|
||||||
{charId: charId, name: skillName, enabled: true}
|
}).forEach(function(effect) {
|
||||||
).forEach(function(proficiency){
|
value = evaluateEffect(charId, effect);
|
||||||
var newProf = proficiency.value;
|
if (effect.operation === "add"){
|
||||||
if (newProf > prof){
|
add += value;
|
||||||
prof = newProf;
|
} else if (effect.operation === "mul"){
|
||||||
|
mul *= value;
|
||||||
|
} else if (effect.operation === "min"){
|
||||||
|
if (value > min) min = value;
|
||||||
|
} else if (effect.operation === "max"){
|
||||||
|
if (value < max) max = value;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
return prof;
|
var result = (mod + add) * mul;
|
||||||
},
|
if (result < min) result = min;
|
||||||
|
if (result > max) result = max;
|
||||||
|
|
||||||
passiveSkill: function(skillName){
|
return Math.floor(result);
|
||||||
if (_.isString(skillName)){
|
})),
|
||||||
var skill = this.getField(skillName);
|
proficiency: memoize(function(charId, skillName){
|
||||||
}
|
//return largest value in proficiency array
|
||||||
var charId = this._id;
|
var prof = Proficiencies.findOne(
|
||||||
var mod = +this.skillMod(skillName);
|
{charId: charId, name: skillName, enabled: true},
|
||||||
|
{sort: {value: -1}}
|
||||||
|
);
|
||||||
|
return prof && prof.value || 0;
|
||||||
|
}),
|
||||||
|
passiveSkill: memoize(function(charId, skillName){
|
||||||
|
var skill = Characters.calculate.getField(charId, skillName);
|
||||||
|
var mod = +Characters.calculate.skillMod(charId, skillName);
|
||||||
var value = 10 + mod;
|
var value = 10 + mod;
|
||||||
Effects.find(
|
Effects.find(
|
||||||
{charId: charId, stat: skillName, enabled: true, operation: "passiveAdd"}
|
{charId: charId, stat: skillName, enabled: true, operation: "passiveAdd"}
|
||||||
).forEach(function(effect){
|
).forEach(function(effect){
|
||||||
value += evaluateEffect(charId, effect);
|
value += evaluateEffect(charId, effect);
|
||||||
});
|
});
|
||||||
return value;
|
var advantage = Characters.calculate.advantage(charId, skillName);
|
||||||
//TODO decide whether (dis)advantage gives (-)+5 to passive checks
|
value += 5 * advantage;
|
||||||
},
|
return Math.floor(value);
|
||||||
|
}),
|
||||||
advantage: function(skillName){
|
advantage: memoize(function(charId, skillName){
|
||||||
var charId = this._id;
|
|
||||||
var advantage = Effects.find(
|
var advantage = Effects.find(
|
||||||
{charId: charId, stat: skillName, enabled: true, operation: "advantage"}
|
{charId: charId, stat: skillName, enabled: true, operation: "advantage"}
|
||||||
).count();
|
).count();
|
||||||
@@ -351,19 +404,18 @@ Characters.helpers({
|
|||||||
if (advantage && !disadvantage) return 1;
|
if (advantage && !disadvantage) return 1;
|
||||||
if (disadvantage && !advantage) return -1;
|
if (disadvantage && !advantage) return -1;
|
||||||
return 0;
|
return 0;
|
||||||
|
}),
|
||||||
|
abilityMod: function(charId, attribute){
|
||||||
|
return getMod(
|
||||||
|
Characters.calculate.attributeValue(charId, attribute)
|
||||||
|
);
|
||||||
},
|
},
|
||||||
|
passiveAbility: function(charId, attribute){
|
||||||
abilityMod: function(attribute){
|
var mod = +getMod(Characters.calculate.attributeValue(charId, attribute));
|
||||||
return signedString(getMod(this.attributeValue(attribute)));
|
|
||||||
},
|
|
||||||
|
|
||||||
passiveAbility: function(attribute){
|
|
||||||
var mod = +getMod(this.attributeValue(attribute));
|
|
||||||
return 10 + mod;
|
return 10 + mod;
|
||||||
},
|
},
|
||||||
|
xpLevel: function(charId){
|
||||||
xpLevel: function(){
|
var xp = Characters.calculate.experience(charId);
|
||||||
var xp = this.experience();
|
|
||||||
for (var i = 0; i < 19; i++){
|
for (var i = 0; i < 19; i++){
|
||||||
if (xp < XP_TABLE[i]){
|
if (xp < XP_TABLE[i]){
|
||||||
return i;
|
return i;
|
||||||
@@ -372,30 +424,103 @@ Characters.helpers({
|
|||||||
if (xp > 355000) return 20;
|
if (xp > 355000) return 20;
|
||||||
return 0;
|
return 0;
|
||||||
},
|
},
|
||||||
|
level: memoize(function(charId){
|
||||||
level: function(){
|
|
||||||
var level = 0;
|
var level = 0;
|
||||||
Classes.find({charId: this._id}).forEach(function(cls){
|
Classes.find({charId: charId}).forEach(function(cls){
|
||||||
level += cls.level;
|
level += cls.level;
|
||||||
});
|
});
|
||||||
return level;
|
return level;
|
||||||
},
|
}),
|
||||||
|
experience: memoize(function(charId){
|
||||||
experience: function(){
|
|
||||||
var xp = 0;
|
var xp = 0;
|
||||||
Experiences.find(
|
Experiences.find(
|
||||||
{charId: this._id},
|
{charId: charId},
|
||||||
{fields: {value: 1}}
|
{fields: {value: 1}}
|
||||||
).forEach(function(e){
|
).forEach(function(e){
|
||||||
xp += e.value;
|
xp += e.value;
|
||||||
});
|
});
|
||||||
return xp;
|
return xp;
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
|
||||||
|
var depreciated = function() {
|
||||||
|
//var err = new Error("this function has been depreciated");
|
||||||
|
var name = "";
|
||||||
|
if (Template.instance()){
|
||||||
|
name = Template.instance().view.name;
|
||||||
|
}
|
||||||
|
var logString = "this function has been depreciated \n";
|
||||||
|
if (name){
|
||||||
|
logString += "View: " + name + "\n\n";
|
||||||
|
}
|
||||||
|
//logString += err.stack + "\n\n---------------------\n\n";
|
||||||
|
console.log(logString);
|
||||||
|
};
|
||||||
|
|
||||||
|
//functions and calculated values.
|
||||||
|
//These functions can only rely on this._id since no other
|
||||||
|
//field is likely to be attached to all returned characters
|
||||||
|
Characters.helpers({
|
||||||
|
//returns the value stored in the field requested
|
||||||
|
//will set up dependencies on just that field
|
||||||
|
getField : function(fieldName){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.getField(this._id, fieldName);
|
||||||
|
},
|
||||||
|
//returns the value of a field
|
||||||
|
fieldValue : function(fieldName){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.fieldValue(this._id, fieldName);
|
||||||
|
},
|
||||||
|
attributeValue: function(attributeName){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.attributeValue(this._id, attributeName);
|
||||||
|
},
|
||||||
|
attributeBase: function(attributeName){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.attributeBase(this._id, attributeName);
|
||||||
|
},
|
||||||
|
skillMod: function(skillName){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.skillMod(this._id, skillName);
|
||||||
|
},
|
||||||
|
proficiency: function(skillName){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.proficiency(this._id, skillName);
|
||||||
|
},
|
||||||
|
passiveSkill: function(skillName){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.passiveSkill(this._id, skillName);
|
||||||
|
},
|
||||||
|
advantage: function(skillName){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.advantage(this._id, skillName);
|
||||||
|
},
|
||||||
|
abilityMod: function(attribute){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.abilityMod(this._id, attribute);
|
||||||
|
},
|
||||||
|
passiveAbility: function(attribute){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.passiveAbility(this._id, attribute);
|
||||||
|
},
|
||||||
|
xpLevel: function(){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.xpLevel(this._id);
|
||||||
|
},
|
||||||
|
level: function(){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.level(this._id);
|
||||||
|
},
|
||||||
|
experience: function(){
|
||||||
|
depreciated();
|
||||||
|
return Characters.calculate.experience(this._id);
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
//clean up all data related to that character before removing it
|
//clean up all data related to that character before removing it
|
||||||
Characters.after.remove(function(userId, character) {
|
if (Meteor.isServer){
|
||||||
if (Meteor.isServer){
|
Characters.after.remove(function(userId, character) {
|
||||||
Actions .remove({charId: character._id});
|
Actions .remove({charId: character._id});
|
||||||
Attacks .remove({charId: character._id});
|
Attacks .remove({charId: character._id});
|
||||||
Buffs .remove({charId: character._id});
|
Buffs .remove({charId: character._id});
|
||||||
@@ -408,8 +533,8 @@ Characters.after.remove(function(userId, character) {
|
|||||||
SpellLists .remove({charId: character._id});
|
SpellLists .remove({charId: character._id});
|
||||||
Items .remove({charId: character._id});
|
Items .remove({charId: character._id});
|
||||||
Containers .remove({charId: character._id});
|
Containers .remove({charId: character._id});
|
||||||
}
|
});
|
||||||
});
|
}
|
||||||
|
|
||||||
Characters.allow({
|
Characters.allow({
|
||||||
insert: function(userId, doc) {
|
insert: function(userId, doc) {
|
||||||
|
|||||||
@@ -107,6 +107,18 @@ if (Meteor.isServer) Characters.after.insert(function(userId, char) {
|
|||||||
group: "Inate",
|
group: "Inate",
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
Effects.insert({
|
||||||
|
charId: char._id,
|
||||||
|
name: "Natural Carrying Capacity",
|
||||||
|
stat: "carryMultiplier",
|
||||||
|
operation: "base",
|
||||||
|
value: "1",
|
||||||
|
parent: {
|
||||||
|
id: char._id,
|
||||||
|
collection: "Characters",
|
||||||
|
group: "Inate",
|
||||||
|
},
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
Effects.attachBehaviour("softRemovable");
|
Effects.attachBehaviour("softRemovable");
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ Schemas.Proficiency = new SimpleSchema({
|
|||||||
Proficiencies.attachSchema(Schemas.Proficiency);
|
Proficiencies.attachSchema(Schemas.Proficiency);
|
||||||
|
|
||||||
Proficiencies.attachBehaviour("softRemovable");
|
Proficiencies.attachBehaviour("softRemovable");
|
||||||
makeChild(Proficiencies);
|
makeChild(Proficiencies, ["enabled"]);
|
||||||
|
|
||||||
Proficiencies.allow(CHARACTER_SUBSCHEMA_ALLOW);
|
Proficiencies.allow(CHARACTER_SUBSCHEMA_ALLOW);
|
||||||
Proficiencies.deny(CHARACTER_SUBSCHEMA_DENY);
|
Proficiencies.deny(CHARACTER_SUBSCHEMA_DENY);
|
||||||
|
|||||||
@@ -62,6 +62,23 @@ Spells.attachSchema(Schemas.Spell);
|
|||||||
|
|
||||||
Spells.attachBehaviour("softRemovable");
|
Spells.attachBehaviour("softRemovable");
|
||||||
makeChild(Spells); //children of spell lists
|
makeChild(Spells); //children of spell lists
|
||||||
|
makeParent(Spells, ["name", "enabled"]); //parents of attacks
|
||||||
|
|
||||||
|
Spells.after.update(function (userId, spell, fieldNames) {
|
||||||
|
//Update prepared state of spell and child attacks to be enabled or not
|
||||||
|
if (_.contains(fieldNames, "prepared")) {
|
||||||
|
var childAttacks = Attacks.find({"parent.id": spell._id}).fetch();
|
||||||
|
if (spell.prepared === "unprepared") {
|
||||||
|
_.each(childAttacks, function(attack){
|
||||||
|
Attacks.update(attack._id, {$set: {enabled: false}});
|
||||||
|
});
|
||||||
|
} else if (spell.prepared === "prepared" || "always") {
|
||||||
|
_.each(childAttacks, function(attack){
|
||||||
|
Attacks.update(attack._id, {$set: {enabled: true}});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
Spells.allow(CHARACTER_SUBSCHEMA_ALLOW);
|
Spells.allow(CHARACTER_SUBSCHEMA_ALLOW);
|
||||||
Spells.deny(CHARACTER_SUBSCHEMA_DENY);
|
Spells.deny(CHARACTER_SUBSCHEMA_DENY);
|
||||||
|
|||||||
@@ -3,10 +3,6 @@
|
|||||||
* Damage, healing and resource cost/recovery are all adjustments
|
* Damage, healing and resource cost/recovery are all adjustments
|
||||||
*/
|
*/
|
||||||
Schemas.Adjustment = new SimpleSchema({
|
Schemas.Adjustment = new SimpleSchema({
|
||||||
name: {
|
|
||||||
type: String,
|
|
||||||
optional: true,
|
|
||||||
},
|
|
||||||
//which stat the adjustment is applied to
|
//which stat the adjustment is applied to
|
||||||
stat: {
|
stat: {
|
||||||
type: String,
|
type: String,
|
||||||
|
|||||||
@@ -3,9 +3,9 @@ TemporaryHitPoints = new Mongo.Collection("temporaryHitPoints");
|
|||||||
Schemas.TemporaryHitPoints = new SimpleSchema({
|
Schemas.TemporaryHitPoints = new SimpleSchema({
|
||||||
charId: {type: String, regEx: SimpleSchema.RegEx.Id},
|
charId: {type: String, regEx: SimpleSchema.RegEx.Id},
|
||||||
name: {type: String, optional: true},
|
name: {type: String, optional: true},
|
||||||
maximum: {type: Number, defaultValue: 0},
|
maximum: {type: Number, defaultValue: 0, min: 0, max: 500},
|
||||||
used: {type: Number, defaultValue: 0},
|
used: {type: Number, defaultValue: 0, min: 0, max: 500},
|
||||||
deleteOnZero:{type: Boolean, defaultValue: true},
|
deleteOnZero:{type: Boolean, defaultValue: false},
|
||||||
dateAdded: {
|
dateAdded: {
|
||||||
type: Date,
|
type: Date,
|
||||||
autoValue: function() {
|
autoValue: function() {
|
||||||
|
|||||||
@@ -45,7 +45,26 @@ Meteor.methods({
|
|||||||
metaData: Object,
|
metaData: Object,
|
||||||
});
|
});
|
||||||
report.owner = this.userId;
|
report.owner = this.userId;
|
||||||
Reports.insert(report);
|
var id = Reports.insert(report);
|
||||||
|
var user = Meteor.users.findOne(this.userId);
|
||||||
|
var sender = user &&
|
||||||
|
user.emails &&
|
||||||
|
user.emails[0] &&
|
||||||
|
user.emails[0].address ||
|
||||||
|
user.services &&
|
||||||
|
user.services.google &&
|
||||||
|
user.services.google.email ||
|
||||||
|
"reports@dicecloud.com";
|
||||||
|
var bodyText = "Report ID: " + id +
|
||||||
|
"\nSeverity: " + report.severity +
|
||||||
|
"\nType: " + report.type +
|
||||||
|
"\n\n" + report.description;
|
||||||
|
Email.send({
|
||||||
|
from: sender,
|
||||||
|
to: "stefan.zermatten@gmail.com",
|
||||||
|
subject: "DiceCloud feedback - " + report.title,
|
||||||
|
text: bodyText,
|
||||||
|
});
|
||||||
},
|
},
|
||||||
deleteReport: function(id) {
|
deleteReport: function(id) {
|
||||||
var user = Meteor.users.findOne(this.userId);
|
var user = Meteor.users.findOne(this.userId);
|
||||||
|
|||||||
@@ -1,20 +1,13 @@
|
|||||||
Router.configure({
|
Router.configure({
|
||||||
loadingTemplate: "loading",
|
loadingTemplate: "loading",
|
||||||
layoutTemplate: "layout",
|
layoutTemplate: "layout",
|
||||||
|
trackPageView: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
Router.plugin("ensureSignedIn", {
|
Router.plugin("ensureSignedIn", {
|
||||||
except: [
|
only: [
|
||||||
"home",
|
"profile",
|
||||||
"atSignIn",
|
"characterList",
|
||||||
"atSignUp",
|
|
||||||
"atForgotPassword",
|
|
||||||
"atResetPwd",
|
|
||||||
"atEnrollAccount",
|
|
||||||
"atVerifyEmail",
|
|
||||||
"atResendVerificationEmail",
|
|
||||||
"loginButtons",
|
|
||||||
"notFound",
|
|
||||||
]
|
]
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -31,11 +24,11 @@ Router.map(function() {
|
|||||||
this.route("characterList", {
|
this.route("characterList", {
|
||||||
path: "/characterList",
|
path: "/characterList",
|
||||||
waitOn: function(){
|
waitOn: function(){
|
||||||
return Meteor.subscribe("characterList", Meteor.userId());
|
return subsManager.subscribe("characterList", Meteor.userId());
|
||||||
},
|
},
|
||||||
data: {
|
data: {
|
||||||
characters: function(){
|
characters: function(){
|
||||||
return Characters.find({}, {fields: {_id: 1}});
|
return Characters.find({}, {fields: {_id: 1}, sort: {name: 1}});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
onAfterAction: function() {
|
onAfterAction: function() {
|
||||||
@@ -47,7 +40,7 @@ Router.map(function() {
|
|||||||
path: "/character/:_id",
|
path: "/character/:_id",
|
||||||
waitOn: function(){
|
waitOn: function(){
|
||||||
return [
|
return [
|
||||||
Meteor.subscribe("singleCharacter", this.params._id, Meteor.userId()),
|
subsManager.subscribe("singleCharacter", this.params._id, Meteor.userId()),
|
||||||
];
|
];
|
||||||
},
|
},
|
||||||
data: function() {
|
data: function() {
|
||||||
@@ -64,6 +57,12 @@ Router.map(function() {
|
|||||||
document.title = name;
|
document.title = name;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
//analytics
|
||||||
|
trackPageView: false,
|
||||||
|
onRun: function() {
|
||||||
|
window.ga && window.ga("send", "pageview", "/character");
|
||||||
|
this.next();
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
this.route("loading", {
|
this.route("loading", {
|
||||||
@@ -81,7 +80,7 @@ Router.map(function() {
|
|||||||
name: "changeLog",
|
name: "changeLog",
|
||||||
waitOn: function() {
|
waitOn: function() {
|
||||||
return [
|
return [
|
||||||
Meteor.subscribe("changeLog"),
|
subsManager.subscribe("changeLog"),
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
data: {
|
data: {
|
||||||
@@ -93,4 +92,11 @@ Router.map(function() {
|
|||||||
document.title = appName;
|
document.title = appName;
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
this.route("/guide", {
|
||||||
|
name: "guide",
|
||||||
|
onAfterAction: function() {
|
||||||
|
document.title = appName;
|
||||||
|
},
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -77,10 +77,10 @@ this.GlobalUI = (function() {
|
|||||||
|
|
||||||
var throttleBack = _.throttle(function() {
|
var throttleBack = _.throttle(function() {
|
||||||
history.back();
|
history.back();
|
||||||
}, 800, {trailing: false});
|
}, 100, {trailing: false});
|
||||||
|
|
||||||
GlobalUI.closeDetail = function() {
|
GlobalUI.closeDetail = function() {
|
||||||
if (!!(window.history && window.history.pushState)) {
|
if (window.history && history.pushState && history.state.detail === "opened") {
|
||||||
throttleBack();
|
throttleBack();
|
||||||
} else {
|
} else {
|
||||||
Session.set("global.ui.detailShow", false);
|
Session.set("global.ui.detailShow", false);
|
||||||
|
|||||||
@@ -1,6 +1,10 @@
|
|||||||
Template.registerHelper("canEditCharacter", function(charId) {
|
Template.registerHelper("canEditCharacter", function(charId) {
|
||||||
|
return canEditCharacter(charId);
|
||||||
|
});
|
||||||
|
|
||||||
|
canEditCharacter = function(charId) {
|
||||||
var char = Characters.findOne(charId)
|
var char = Characters.findOne(charId)
|
||||||
var userId = Meteor.userId();
|
var userId = Meteor.userId();
|
||||||
return char.owner === userId ||
|
return char.owner === userId ||
|
||||||
_.contains(char.writers, userId);
|
_.contains(char.writers, userId);
|
||||||
});
|
};
|
||||||
|
|||||||
@@ -20,6 +20,11 @@ openParentDialog = function(parent, charId, heroId) {
|
|||||||
template: "itemDialog",
|
template: "itemDialog",
|
||||||
data: {itemId: parent.id},
|
data: {itemId: parent.id},
|
||||||
};
|
};
|
||||||
|
} else if (parent.collection === "Spells") {
|
||||||
|
detail = {
|
||||||
|
template: "spellDialog",
|
||||||
|
data: {spellId: parent.id},
|
||||||
|
};
|
||||||
}
|
}
|
||||||
detail.heroId = heroId;
|
detail.heroId = heroId;
|
||||||
detail.charId = charId;
|
detail.charId = charId;
|
||||||
|
|||||||
@@ -1,25 +1,27 @@
|
|||||||
Template.registerHelper("valueString", function(value) {
|
Template.registerHelper("valueString", function(value) {
|
||||||
|
var intValue = Math.round(value * 100);
|
||||||
|
var cp = intValue % 10;
|
||||||
|
intValue -= cp;
|
||||||
|
cp = Math.round(cp);
|
||||||
|
sp = intValue % 100;
|
||||||
|
intValue -= sp;
|
||||||
|
sp = Math.round(sp / 10)
|
||||||
|
gp = Math.floor(value);
|
||||||
|
|
||||||
var resultArray = [];
|
var resultArray = [];
|
||||||
//sp
|
|
||||||
var gp = Math.floor(value);
|
|
||||||
if (gp > 0) {
|
if (gp > 0) {
|
||||||
resultArray.push(gp + "gp");
|
resultArray.push(gp + "gp");
|
||||||
}
|
}
|
||||||
//sp
|
|
||||||
var sp = Math.floor(10 * (value % 1));
|
|
||||||
if (sp > 0) {
|
if (sp > 0) {
|
||||||
resultArray.push(sp + "sp");
|
resultArray.push(sp + "sp");
|
||||||
}
|
}
|
||||||
//cp
|
|
||||||
var cp = 10 * ((value * 10) % 1);
|
|
||||||
cp = Math.round(cp * 1000) / 1000;
|
|
||||||
if (cp > 0) {
|
if (cp > 0) {
|
||||||
resultArray.push(cp + "cp");
|
resultArray.push(cp + "cp");
|
||||||
}
|
}
|
||||||
|
|
||||||
//build string with correct spacing
|
//build string with correct spacing
|
||||||
var result = "";
|
var result = "";
|
||||||
for (var i = 0; i < resultArray.length; i++) {
|
for (var i = 0, l = resultArray.length; i < l; i++) {
|
||||||
//add a space between values
|
//add a space between values
|
||||||
if (i !== 0) {
|
if (i !== 0) {
|
||||||
result += " ";
|
result += " ";
|
||||||
|
|||||||
@@ -3,17 +3,24 @@
|
|||||||
$thickColumnWidth: 304px;
|
$thickColumnWidth: 304px;
|
||||||
$thinColumnWidth: 240px;
|
$thinColumnWidth: 240px;
|
||||||
|
|
||||||
//Column layouts of cards
|
//Column layout
|
||||||
.column-container {
|
.column-container {
|
||||||
@include column-fill(balance);
|
@include column-fill(balance);
|
||||||
@include column-gap(8px);
|
@include column-gap(0px);
|
||||||
@include column-width($thickColumnWidth);
|
@include column-width($thickColumnWidth);
|
||||||
padding: 8px;
|
padding: 4px;
|
||||||
|
|
||||||
&.thin-columns {
|
&.thin-columns {
|
||||||
@include column-count(4);
|
@include column-count(4);
|
||||||
@include column-width($thinColumnWidth);
|
@include column-width($thinColumnWidth);
|
||||||
}
|
}
|
||||||
|
& > div {
|
||||||
|
padding: 4px;
|
||||||
|
//stop divs breaking over multiple columns
|
||||||
|
-webkit-column-break-inside: avoid;
|
||||||
|
page-break-inside: avoid;
|
||||||
|
break-inside: avoid;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//Cards
|
//Cards
|
||||||
@@ -21,20 +28,6 @@ $thinColumnWidth: 240px;
|
|||||||
background: white;
|
background: white;
|
||||||
border-radius: 2px;
|
border-radius: 2px;
|
||||||
|
|
||||||
.column-container & {
|
|
||||||
margin-bottom: 8px;
|
|
||||||
width: 100%;
|
|
||||||
//hack to stop flickering
|
|
||||||
-webkit-backface-visibility: hidden;
|
|
||||||
-webkit-transform: translateX(0);
|
|
||||||
//stop breaking over column divide
|
|
||||||
-webkit-column-break-inside: avoid;
|
|
||||||
page-break-inside: avoid;
|
|
||||||
break-inside: avoid;
|
|
||||||
//Fixes extra margin at top of columns
|
|
||||||
display: inline-block;
|
|
||||||
}
|
|
||||||
|
|
||||||
.top {
|
.top {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
padding: 16px;
|
padding: 16px;
|
||||||
@@ -89,3 +82,8 @@ $thinColumnWidth: 240px;
|
|||||||
border-radius: 0 2px 2px 0;
|
border-radius: 0 2px 2px 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* undo pointer cursor on detail box heading */
|
||||||
|
#globalDetail.card .top {
|
||||||
|
cursor: auto;
|
||||||
|
}
|
||||||
|
|||||||
@@ -70,8 +70,8 @@
|
|||||||
background-color: #9E9E9E;
|
background-color: #9E9E9E;
|
||||||
}
|
}
|
||||||
|
|
||||||
.blue-grey {
|
.app-grey {
|
||||||
background-color: #607D8B;
|
background-color: #424242;
|
||||||
}
|
}
|
||||||
|
|
||||||
.white {
|
.white {
|
||||||
|
|||||||
@@ -19,6 +19,21 @@ body {
|
|||||||
background-color: #E0E0E0;
|
background-color: #E0E0E0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//fix tabs and core-toolbar having box shadow
|
||||||
|
core-toolbar {
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
//give drawer panel a shadow always
|
||||||
|
core-header-panel[drawer] {
|
||||||
|
box-shadow: 2px 0px 5px 0px rgba(0,0,0,0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
//Paragraphs
|
||||||
|
p {
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
//Horizontal rule
|
//Horizontal rule
|
||||||
hr {
|
hr {
|
||||||
background-color: #444;
|
background-color: #444;
|
||||||
@@ -27,7 +42,7 @@ hr {
|
|||||||
color: #444;
|
color: #444;
|
||||||
height: 1px;
|
height: 1px;
|
||||||
line-height: 0;
|
line-height: 0;
|
||||||
margin: 16px -16px;
|
margin: 16px 0;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,5 @@
|
|||||||
td {
|
td {
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
&:nth-child(1) {
|
|
||||||
min-width: 100px;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.strengthTable{
|
.strengthTable{
|
||||||
@@ -15,4 +12,10 @@ td {
|
|||||||
width: 250px;
|
width: 250px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.summaryTable {
|
||||||
|
&:nth-child(3){
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -3,26 +3,26 @@
|
|||||||
<div layout vertical flex>
|
<div layout vertical flex>
|
||||||
<div layout horizontal>
|
<div layout horizontal>
|
||||||
<!--attackBonus-->
|
<!--attackBonus-->
|
||||||
<paper-input id="attackBonusInput"
|
<paper-input class="attackBonusInput"
|
||||||
label="Attack Bonus"
|
label="Attack Bonus"
|
||||||
floatinglabel
|
floatinglabel
|
||||||
value={{attackBonus}}
|
value={{attackBonus}}
|
||||||
flex></paper-input>
|
flex></paper-input>
|
||||||
<!--details-->
|
<!--details-->
|
||||||
<paper-input id="detailInput"
|
<paper-input class="detailInput"
|
||||||
label="Details"
|
label="Details"
|
||||||
floatinglabel
|
floatinglabel
|
||||||
value={{details}}></paper-input>
|
value={{details}}></paper-input>
|
||||||
</div>
|
</div>
|
||||||
<div layout horizontal>
|
<div layout horizontal>
|
||||||
<!--damageBonus-->
|
<!--damageBonus-->
|
||||||
<paper-input id="damageInput"
|
<paper-input class="damageInput"
|
||||||
label="Damage"
|
label="Damage"
|
||||||
floatinglabel
|
floatinglabel
|
||||||
value={{damage}}
|
value={{damage}}
|
||||||
flex></paper-input>
|
flex></paper-input>
|
||||||
<!--DamageType-->
|
<!--DamageType-->
|
||||||
<paper-dropdown-menu id="damageTypeDropdown" label="Damage Type">
|
<paper-dropdown-menu class="damageTypeDropdown" label="Damage Type">
|
||||||
<paper-dropdown layered class="dropdown">
|
<paper-dropdown layered class="dropdown">
|
||||||
<core-menu class="menu" selected={{damageType}}>
|
<core-menu class="menu" selected={{damageType}}>
|
||||||
{{#each damageTypes}}
|
{{#each damageTypes}}
|
||||||
@@ -34,6 +34,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<!--Delete Button-->
|
<!--Delete Button-->
|
||||||
<paper-icon-button id="deleteAttack" role="button" tabindex="0" icon="delete" aria-label="Delete"></paper-icon-button>
|
<paper-icon-button class="deleteAttack" role="button" tabindex="0" icon="delete" aria-label="Delete"></paper-icon-button>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -15,23 +15,23 @@ var damageTypes = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
Template.attackEdit.events({
|
Template.attackEdit.events({
|
||||||
"tap #deleteAttack": function(event, instance) {
|
"tap .deleteAttack": function(event, instance) {
|
||||||
Attacks.softRemoveNode(this._id);
|
Attacks.softRemoveNode(this._id);
|
||||||
GlobalUI.deletedToast(this._id, "Attacks", "Attack");
|
GlobalUI.deletedToast(this._id, "Attacks", "Attack");
|
||||||
},
|
},
|
||||||
"change #attackBonusInput": function(event) {
|
"change .attackBonusInput": function(event) {
|
||||||
var value = event.currentTarget.value;
|
var value = event.currentTarget.value;
|
||||||
Attacks.update(this._id, {$set: {attackBonus: value}});
|
Attacks.update(this._id, {$set: {attackBonus: value}});
|
||||||
},
|
},
|
||||||
"change #damageInput": function(event) {
|
"change .damageInput": function(event) {
|
||||||
var value = event.currentTarget.value;
|
var value = event.currentTarget.value;
|
||||||
Attacks.update(this._id, {$set: {damage: value}});
|
Attacks.update(this._id, {$set: {damage: value}});
|
||||||
},
|
},
|
||||||
"change #detailInput": function(event) {
|
"change .detailInput": function(event) {
|
||||||
var value = event.currentTarget.value;
|
var value = event.currentTarget.value;
|
||||||
Attacks.update(this._id, {$set: {details: value}});
|
Attacks.update(this._id, {$set: {details: value}});
|
||||||
},
|
},
|
||||||
"core-select #damageTypeDropdown": function(event) {
|
"core-select .damageTypeDropdown": function(event) {
|
||||||
var detail = event.originalEvent.detail;
|
var detail = event.originalEvent.detail;
|
||||||
if (!detail.isSelected) return;
|
if (!detail.isSelected) return;
|
||||||
var value = detail.item.getAttribute("name");
|
var value = detail.item.getAttribute("name");
|
||||||
|
|||||||
@@ -0,0 +1,15 @@
|
|||||||
|
<template name="buffDialog">
|
||||||
|
{{#with buff}}
|
||||||
|
{{#baseDialog title=name class=colorClass hideEdit=true}}
|
||||||
|
{{> buffDetails}}
|
||||||
|
{{/baseDialog}}
|
||||||
|
{{/with}}
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template name="buffDetails">
|
||||||
|
{{#if description}}
|
||||||
|
<div class="pre-wrap">{{evaluateString charId description}}</div>
|
||||||
|
{{/if}}
|
||||||
|
|
||||||
|
{{> effectsViewList charId=charId parentId=_id}}
|
||||||
|
</template>
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
Template.buffDialog.helpers({
|
||||||
|
buff: function(){
|
||||||
|
return Buffs.findOne(this.buffId);
|
||||||
|
},
|
||||||
|
});
|
||||||
@@ -1,3 +1,27 @@
|
|||||||
<template name="characterSettings">
|
<template name="characterSettings">
|
||||||
|
{{#with character}}
|
||||||
|
<div style="height: 100px;">
|
||||||
|
<table style="width: 100%;">
|
||||||
|
<tr>
|
||||||
|
<td>Hide Spells tab</td>
|
||||||
|
<td>
|
||||||
|
<paper-toggle-button id="hideSpellcasting"
|
||||||
|
checked={{settings.hideSpellcasting}}
|
||||||
|
touch-action="pan-y">
|
||||||
|
</paper-toggle-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>Use variant encumbrance</td>
|
||||||
|
<td>
|
||||||
|
<paper-toggle-button id="variantEncumbrance"
|
||||||
|
checked={{settings.useVariantEncumbrance}}
|
||||||
|
touch-action="pan-y">
|
||||||
|
</paper-toggle-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
{{/with}}
|
||||||
|
<paper-button id="doneButton" affirmative> Done </paper-button>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -1,3 +1,26 @@
|
|||||||
Template.characterSettings.events({
|
Template.characterSettings.helpers({
|
||||||
|
character: function() {
|
||||||
|
return Characters.findOne(this._id, {fields: {settings: 1}});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Template.characterSettings.events({
|
||||||
|
"change #variantEncumbrance": function(event, instance){
|
||||||
|
var value = instance.find("#variantEncumbrance").checked;
|
||||||
|
if (this.settings.useVariantEncumbrance !== value){
|
||||||
|
Characters.update(
|
||||||
|
this._id,
|
||||||
|
{$set: {"settings.useVariantEncumbrance": value}}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"change #hideSpellcasting": function(event, instance){
|
||||||
|
var value = instance.find("#hideSpellcasting").checked;
|
||||||
|
if (this.settings.hideSpellcasting !== value){
|
||||||
|
Characters.update(
|
||||||
|
this._id,
|
||||||
|
{$set: {"settings.hideSpellcasting": value}}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,5 +1,17 @@
|
|||||||
<template name="shareDialog">
|
<template name="shareDialog">
|
||||||
<div style="width: 360px;">
|
<div style="width: 360px;">
|
||||||
|
<div layout horizontal center>
|
||||||
|
<div>Who can view this character: </div>
|
||||||
|
<paper-dropdown-menu class="visibilityDropdown"
|
||||||
|
label="Visibility">
|
||||||
|
<paper-dropdown layered class="dropdown">
|
||||||
|
<core-menu class="menu visibilityMenu" selected={{viewPermission}}>
|
||||||
|
<paper-item name="whitelist">Only people I share with</paper-item>
|
||||||
|
<paper-item name="public">Anyone with link</paper-item>
|
||||||
|
</core-menu>
|
||||||
|
</paper-dropdown>
|
||||||
|
</paper-dropdown-menu>
|
||||||
|
</div>
|
||||||
<div>
|
<div>
|
||||||
{{#if readers.count}}
|
{{#if readers.count}}
|
||||||
<div style="font-weight: 500;">
|
<div style="font-weight: 500;">
|
||||||
@@ -7,7 +19,7 @@
|
|||||||
</div>
|
</div>
|
||||||
{{#each readers}}
|
{{#each readers}}
|
||||||
<div layout horizontal center>
|
<div layout horizontal center>
|
||||||
<div flex>{{username}}</div>
|
<div flex>{{getUserName}}</div>
|
||||||
<paper-icon-button class="deleteShare" icon="delete"></paper-icon-button>
|
<paper-icon-button class="deleteShare" icon="delete"></paper-icon-button>
|
||||||
</div>
|
</div>
|
||||||
{{/each}}
|
{{/each}}
|
||||||
|
|||||||
@@ -3,6 +3,10 @@ Template.shareDialog.onCreated(function(){
|
|||||||
});
|
});
|
||||||
|
|
||||||
Template.shareDialog.helpers({
|
Template.shareDialog.helpers({
|
||||||
|
viewPermission: function() {
|
||||||
|
var char = Characters.findOne(this._id, {fields: {settings: 1}});
|
||||||
|
return char.settings.viewPermission || "whitelist";
|
||||||
|
},
|
||||||
readers: function(){
|
readers: function(){
|
||||||
var char = Characters.findOne(this._id, {fields: {readers: 1}});
|
var char = Characters.findOne(this._id, {fields: {readers: 1}});
|
||||||
return Meteor.users.find({_id: {$in: char.readers}});
|
return Meteor.users.find({_id: {$in: char.readers}});
|
||||||
@@ -19,9 +23,20 @@ Template.shareDialog.helpers({
|
|||||||
return "User not found";
|
return "User not found";
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
getUserName: function() {
|
||||||
|
return this.username || "user: " + this._id;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
Template.shareDialog.events({
|
Template.shareDialog.events({
|
||||||
|
"core-select .visibilityDropdown": function(event){
|
||||||
|
var detail = event.originalEvent.detail;
|
||||||
|
if (!detail.isSelected) return;
|
||||||
|
var value = detail.item.getAttribute("name");
|
||||||
|
var char = Characters.findOne(this._id, {fields: {settings: 1}});
|
||||||
|
if (value == char.settings.viewPermission) return;
|
||||||
|
Characters.update(this._id, {$set: {"settings.viewPermission": value}});
|
||||||
|
},
|
||||||
"input #userNameOrEmailInput":
|
"input #userNameOrEmailInput":
|
||||||
function(event, instance){
|
function(event, instance){
|
||||||
var userName = instance.find("#userNameOrEmailInput").value;
|
var userName = instance.find("#userNameOrEmailInput").value;
|
||||||
|
|||||||
@@ -19,6 +19,9 @@
|
|||||||
<paper-item id="shareCharacter">
|
<paper-item id="shareCharacter">
|
||||||
<core-icon icon="social:share"></core-icon>Share
|
<core-icon icon="social:share"></core-icon>Share
|
||||||
</paper-item>
|
</paper-item>
|
||||||
|
<paper-item id="characterSettings">
|
||||||
|
<core-icon icon="settings"></core-icon>Settings
|
||||||
|
</paper-item>
|
||||||
</core-menu>
|
</core-menu>
|
||||||
</paper-dropdown>
|
</paper-dropdown>
|
||||||
</paper-menu-button>
|
</paper-menu-button>
|
||||||
@@ -28,7 +31,9 @@
|
|||||||
<paper-tab name="stats">Stats</paper-tab>
|
<paper-tab name="stats">Stats</paper-tab>
|
||||||
<paper-tab name="features">Features</paper-tab>
|
<paper-tab name="features">Features</paper-tab>
|
||||||
<paper-tab name="inventory">Inventory</paper-tab>
|
<paper-tab name="inventory">Inventory</paper-tab>
|
||||||
|
{{#unless hideSpellcasting}}
|
||||||
<paper-tab name="spells">Spells</paper-tab>
|
<paper-tab name="spells">Spells</paper-tab>
|
||||||
|
{{/unless}}
|
||||||
<paper-tab name="persona">Persona</paper-tab>
|
<paper-tab name="persona">Persona</paper-tab>
|
||||||
<paper-tab name="journal">Journal</paper-tab>
|
<paper-tab name="journal">Journal</paper-tab>
|
||||||
</paper-tabs>
|
</paper-tabs>
|
||||||
@@ -39,7 +44,9 @@
|
|||||||
<section flex name="stats">{{> stats}}</section>
|
<section flex name="stats">{{> stats}}</section>
|
||||||
<section flex name="features">{{> features}}</section>
|
<section flex name="features">{{> features}}</section>
|
||||||
<section flex name="inventory">{{> inventory}}</section>
|
<section flex name="inventory">{{> inventory}}</section>
|
||||||
|
{{#unless hideSpellcasting}}
|
||||||
<section flex name="spells">{{> spells}}</section>
|
<section flex name="spells">{{> spells}}</section>
|
||||||
|
{{/unless}}
|
||||||
<section flex name="persona">{{> persona}}</section>
|
<section flex name="persona">{{> persona}}</section>
|
||||||
<section flex name="journal">{{> journal}}</section>
|
<section flex name="journal">{{> journal}}</section>
|
||||||
</core-animated-pages>
|
</core-animated-pages>
|
||||||
|
|||||||
@@ -1,6 +1,9 @@
|
|||||||
Template.characterSheet.created = function(){
|
Template.characterSheet.onCreated(function() {
|
||||||
|
//default to the first tab
|
||||||
Session.setDefault(this.data._id + ".selectedTab", "stats");
|
Session.setDefault(this.data._id + ".selectedTab", "stats");
|
||||||
};
|
//watch this character and make sure their encumbrance is updated
|
||||||
|
trackEncumbranceConditions(this.data._id, this);
|
||||||
|
});
|
||||||
|
|
||||||
var setTab = function(charId, tab){
|
var setTab = function(charId, tab){
|
||||||
return Session.set(charId + ".selectedTab", tab);
|
return Session.set(charId + ".selectedTab", tab);
|
||||||
@@ -14,6 +17,10 @@ Template.characterSheet.helpers({
|
|||||||
selectedTab: function(){
|
selectedTab: function(){
|
||||||
return getTab(this._id);
|
return getTab(this._id);
|
||||||
},
|
},
|
||||||
|
hideSpellcasting: function() {
|
||||||
|
var char = Characters.findOne(this._id);
|
||||||
|
return char && char.settings.hideSpellcasting;
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
Template.characterSheet.events({
|
Template.characterSheet.events({
|
||||||
@@ -40,4 +47,11 @@ Template.characterSheet.events({
|
|||||||
template: "shareDialog",
|
template: "shareDialog",
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
"tap #characterSettings": function(event, instance){
|
||||||
|
GlobalUI.showDialog({
|
||||||
|
heading: this.name + " Settings",
|
||||||
|
data: this,
|
||||||
|
template: "characterSettings",
|
||||||
|
});
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,21 +1,13 @@
|
|||||||
body /deep/ #statGroupDropDown {
|
html /deep/ .operationDropDown {
|
||||||
width: 120px;
|
width: 152px;
|
||||||
}
|
}
|
||||||
|
|
||||||
body /deep/ #statDropDown {
|
html /deep/ .statDropDown {
|
||||||
width: 120px;
|
width: 152px;
|
||||||
}
|
}
|
||||||
|
|
||||||
body /deep/ #operationDropDown {
|
html /deep/ .damageMultiplierDropDown {
|
||||||
width: 100px;
|
width: 152px;
|
||||||
}
|
|
||||||
|
|
||||||
body /deep/ #damageMultiplierDropDown {
|
|
||||||
width: 120px;
|
|
||||||
}
|
|
||||||
|
|
||||||
body /deep/ #proficiencyDropDown {
|
|
||||||
width: 120px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
html /deep/ .effectEdit paper-input {
|
html /deep/ .effectEdit paper-input {
|
||||||
@@ -24,6 +16,7 @@ html /deep/ .effectEdit paper-input {
|
|||||||
}
|
}
|
||||||
|
|
||||||
html /deep/ .effectEdit {
|
html /deep/ .effectEdit {
|
||||||
|
height: 64px;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: flex-end;
|
align-items: flex-end;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,27 +1,23 @@
|
|||||||
<template name="effectEdit">
|
<template name="effectEdit">
|
||||||
<div class="effectEdit" layout horizontal center>
|
<div class="effectEdit" layout horizontal center>
|
||||||
<paper-dropdown-menu class="statGroupDropDown" label="Stat Group" flex>
|
<paper-dropdown-menu class="statDropDown"
|
||||||
<paper-dropdown layered class="dropdown">
|
label="Stat">
|
||||||
<core-menu class="menu statGroupMenu" selected={{selectedStatGroup}}>
|
<paper-dropdown layered
|
||||||
{{#each statGroups}}
|
class="dropdown">
|
||||||
<paper-item class="statGroupSelect" name={{this}}>{{this}}</paper-item>
|
<core-menu class="menu statMenu" selected={{stat}}>
|
||||||
{{/each}}
|
{{#each statGroups}}
|
||||||
</core-menu>
|
<div style="font-weight: bold;
|
||||||
</paper-dropdown>
|
margin-top: 16px;">{{this}}</div>
|
||||||
</paper-dropdown-menu>
|
{{#each stats}}
|
||||||
{{#if stats}}
|
<paper-item name={{stat}}>{{name}}</paper-item>
|
||||||
<paper-dropdown-menu class="statDropDown" label="Stat" flex>
|
{{/each}}
|
||||||
<paper-dropdown layered class="dropdown">
|
|
||||||
<core-menu class="menu statMenu" selected={{stat}} on-tap="onStatMenuTap">
|
|
||||||
{{#each stats}}
|
|
||||||
<paper-item name={{stat}}>{{name}}</paper-item>
|
|
||||||
{{/each}}
|
{{/each}}
|
||||||
</core-menu>
|
</core-menu>
|
||||||
</paper-dropdown>
|
</paper-dropdown>
|
||||||
</paper-dropdown-menu>
|
</paper-dropdown-menu>
|
||||||
{{/if}}
|
|
||||||
{{#if operations}}
|
{{#if operations}}
|
||||||
<paper-dropdown-menu class="operationDropDown" label="Operation" flex>
|
<paper-dropdown-menu class="operationDropDown"
|
||||||
|
label="Operation">
|
||||||
<paper-dropdown layered class="dropdown">
|
<paper-dropdown layered class="dropdown">
|
||||||
<core-menu class="menu operationMenu" selected={{operation}}>
|
<core-menu class="menu operationMenu" selected={{operation}}>
|
||||||
{{#each operations}}
|
{{#each operations}}
|
||||||
@@ -31,24 +27,40 @@
|
|||||||
</paper-dropdown>
|
</paper-dropdown>
|
||||||
</paper-dropdown-menu>
|
</paper-dropdown-menu>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
{{> Template.dynamic template=effectValueTemplate}}
|
{{#if effectValueTemplate}}
|
||||||
<paper-icon-button class="deleteEffect" role="button" tabindex="0" icon="delete" aria-label="Delete"></paper-icon-button>
|
{{> Template.dynamic template=effectValueTemplate}}
|
||||||
|
{{else}}
|
||||||
|
<div flex></div>
|
||||||
|
{{/if}}
|
||||||
|
<paper-icon-button class="deleteEffect"
|
||||||
|
icon="delete">
|
||||||
|
</paper-icon-button>
|
||||||
<br>
|
<br>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template name="regularEffectValue">
|
<template name="regularEffectValue">
|
||||||
<paper-input class="effectValueInput" label="Value" floatinglabel value={{effectValue}} flex></paper-input>
|
<paper-input class="effectValueInput"
|
||||||
|
label="Value"
|
||||||
|
floatinglabel
|
||||||
|
value={{effectValue}}
|
||||||
|
flex
|
||||||
|
style="flex-basis: 100px;">
|
||||||
|
</paper-input>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template name="multiplierEffectValue">
|
<template name="multiplierEffectValue">
|
||||||
<paper-dropdown-menu class="damageMultiplierDropDown" label="Damage Multiplier" flex>
|
<paper-dropdown-menu class="damageMultiplierDropDown"
|
||||||
<paper-dropdown layered class="dropdown">
|
label="Damage Multiplier">
|
||||||
<core-menu class="menu multiplierMenu" selected={{value}}>
|
<paper-dropdown layered
|
||||||
|
class="dropdown">
|
||||||
|
<core-menu class="menu multiplierMenu"
|
||||||
|
selected={{value}}>
|
||||||
<paper-item name="0.5">Resistance</paper-item>
|
<paper-item name="0.5">Resistance</paper-item>
|
||||||
<paper-item name="2">Vulnerability</paper-item>
|
<paper-item name="2">Vulnerability</paper-item>
|
||||||
<paper-item name="0">Immunity</paper-item>
|
<paper-item name="0">Immunity</paper-item>
|
||||||
</core-menu>
|
</core-menu>
|
||||||
</paper-dropdown>
|
</paper-dropdown>
|
||||||
</paper-dropdown-menu>
|
</paper-dropdown-menu>
|
||||||
|
<div flex></div>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -42,6 +42,7 @@ var stats = [
|
|||||||
{stat: "rageDamage", name: "Rage Damage", group: "Stats"},
|
{stat: "rageDamage", name: "Rage Damage", group: "Stats"},
|
||||||
{stat: "expertiseDice", name: "Expertise Dice", group: "Stats"},
|
{stat: "expertiseDice", name: "Expertise Dice", group: "Stats"},
|
||||||
{stat: "superiorityDice", name: "Superiority Dice", group: "Stats"},
|
{stat: "superiorityDice", name: "Superiority Dice", group: "Stats"},
|
||||||
|
{stat: "carryMultiplier", name: "Carry Capacity Multiplier", group: "Stats"},
|
||||||
{stat: "level1SpellSlots", name: "level 1", group: "Spell Slots"},
|
{stat: "level1SpellSlots", name: "level 1", group: "Spell Slots"},
|
||||||
{stat: "level2SpellSlots", name: "level 2", group: "Spell Slots"},
|
{stat: "level2SpellSlots", name: "level 2", group: "Spell Slots"},
|
||||||
{stat: "level3SpellSlots", name: "level 3", group: "Spell Slots"},
|
{stat: "level3SpellSlots", name: "level 3", group: "Spell Slots"},
|
||||||
@@ -93,24 +94,17 @@ var skillOperations = [
|
|||||||
{name: "Conditional Benefit", operation: "conditional"}
|
{name: "Conditional Benefit", operation: "conditional"}
|
||||||
];
|
];
|
||||||
|
|
||||||
Template.effectEdit.created = function(){
|
|
||||||
var statGroup = statsDict[this.data.stat] && statsDict[this.data.stat].group;
|
|
||||||
this.selectedStatGroup = new ReactiveVar(statGroup);
|
|
||||||
};
|
|
||||||
|
|
||||||
Template.effectEdit.helpers({
|
Template.effectEdit.helpers({
|
||||||
selectedStatGroup: function(){
|
|
||||||
return Template.instance().selectedStatGroup.get();
|
|
||||||
},
|
|
||||||
statGroups: function(){
|
statGroups: function(){
|
||||||
return statGroupNames;
|
return statGroupNames;
|
||||||
},
|
},
|
||||||
stats: function(){
|
stats: function(){
|
||||||
var group = Template.instance().selectedStatGroup.get();
|
var group = this;
|
||||||
return statGroups[group];
|
return statGroups[group];
|
||||||
},
|
},
|
||||||
operations: function(){
|
operations: function(){
|
||||||
var group = Template.instance().selectedStatGroup.get();
|
var stat = statsDict[this.stat];
|
||||||
|
var group = stat && stat.group;
|
||||||
if (group === "Weakness/Resistance") return null;
|
if (group === "Weakness/Resistance") return null;
|
||||||
if (group === "Saving Throws" || group === "Skills"){
|
if (group === "Saving Throws" || group === "Skills"){
|
||||||
return skillOperations;
|
return skillOperations;
|
||||||
@@ -120,7 +114,8 @@ Template.effectEdit.helpers({
|
|||||||
},
|
},
|
||||||
effectValueTemplate: function(){
|
effectValueTemplate: function(){
|
||||||
//resistance/vulnerability template
|
//resistance/vulnerability template
|
||||||
var group = Template.instance().selectedStatGroup.get();
|
var stat = statsDict[this.stat];
|
||||||
|
var group = stat && stat.group;
|
||||||
if (group === "Weakness/Resistance") return "multiplierEffectValue";
|
if (group === "Weakness/Resistance") return "multiplierEffectValue";
|
||||||
|
|
||||||
var op = this.operation;
|
var op = this.operation;
|
||||||
@@ -144,25 +139,6 @@ Template.effectEdit.events({
|
|||||||
Effects.softRemoveNode(this._id);
|
Effects.softRemoveNode(this._id);
|
||||||
GlobalUI.deletedToast(this._id, "Effects", "Effect");
|
GlobalUI.deletedToast(this._id, "Effects", "Effect");
|
||||||
},
|
},
|
||||||
"core-select .statGroupDropDown": function(event, instance){
|
|
||||||
var detail = event.originalEvent.detail;
|
|
||||||
if (!detail.isSelected) return;
|
|
||||||
var groupName = detail.item.getAttribute("name");
|
|
||||||
var oldName = Template.instance().selectedStatGroup.get();
|
|
||||||
if (oldName != groupName){
|
|
||||||
instance.selectedStatGroup.set(groupName);
|
|
||||||
if (groupName === "Weakness/Resistance"){
|
|
||||||
Effects.update(this._id, {$set: {
|
|
||||||
value: 0.5,
|
|
||||||
calculation: "",
|
|
||||||
operation: "mul"}, $unset: {stat: ""}});
|
|
||||||
} else {
|
|
||||||
Effects.update(this._id,
|
|
||||||
{$set: {operation: "add"},
|
|
||||||
$unset: {stat: "", value: "", calculation: ""}});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"core-select .statDropDown": function(event){
|
"core-select .statDropDown": function(event){
|
||||||
var detail = event.originalEvent.detail;
|
var detail = event.originalEvent.detail;
|
||||||
if (!detail.isSelected) return;
|
if (!detail.isSelected) return;
|
||||||
|
|||||||
@@ -42,6 +42,7 @@ var stats = {
|
|||||||
"rageDamage":{"name":"Rage Damage"},
|
"rageDamage":{"name":"Rage Damage"},
|
||||||
"expertiseDice":{"name":"Expertise Dice"},
|
"expertiseDice":{"name":"Expertise Dice"},
|
||||||
"superiorityDice":{"name":"Superiority Dice"},
|
"superiorityDice":{"name":"Superiority Dice"},
|
||||||
|
"carryMultiplier": {"name": "Carry Capacity Multiplier"},
|
||||||
"level1SpellSlots":{"name":"level 1 Spell Slots"},
|
"level1SpellSlots":{"name":"level 1 Spell Slots"},
|
||||||
"level2SpellSlots":{"name":"level 2 Spell Slots"},
|
"level2SpellSlots":{"name":"level 2 Spell Slots"},
|
||||||
"level3SpellSlots":{"name":"level 3 Spell Slots"},
|
"level3SpellSlots":{"name":"level 3 Spell Slots"},
|
||||||
@@ -57,40 +58,40 @@ var stats = {
|
|||||||
"d12HitDice":{"name":"d12 Hit Dice"},
|
"d12HitDice":{"name":"d12 Hit Dice"},
|
||||||
"acidMultiplier":{"name":"Acid damage", "group": "Weakness/Resistance"},
|
"acidMultiplier":{"name":"Acid damage", "group": "Weakness/Resistance"},
|
||||||
"bludgeoningMultiplier":{
|
"bludgeoningMultiplier":{
|
||||||
"name":"Bludgeoning damage", "group": "Weakness/Resistance"
|
"name":"Bludgeoning damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
"coldMultiplier":{
|
"coldMultiplier":{
|
||||||
"name":"Cold damage", "group": "Weakness/Resistance"
|
"name":"Cold damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
"fireMultiplier":{
|
"fireMultiplier":{
|
||||||
"name":"Fire damage", "group": "Weakness/Resistance"
|
"name":"Fire damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
"forceMultiplier":{
|
"forceMultiplier":{
|
||||||
"name":"Force damage", "group": "Weakness/Resistance"
|
"name":"Force damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
"lightningMultiplier":{
|
"lightningMultiplier":{
|
||||||
"name":"Lightning damage", "group": "Weakness/Resistance"
|
"name":"Lightning damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
"necroticMultiplier":{
|
"necroticMultiplier":{
|
||||||
"name":"Necrotic damage", "group": "Weakness/Resistance"
|
"name":"Necrotic damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
"piercingMultiplier":{
|
"piercingMultiplier":{
|
||||||
"name":"Piercing damage", "group": "Weakness/Resistance"
|
"name":"Piercing damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
"poisonMultiplier":{
|
"poisonMultiplier":{
|
||||||
"name":"Poison damage", "group": "Weakness/Resistance"
|
"name":"Poison damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
"psychicMultiplier":{
|
"psychicMultiplier":{
|
||||||
"name":"Psychic damage", "group": "Weakness/Resistance"
|
"name":"Psychic damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
"radiantMultiplier":{
|
"radiantMultiplier":{
|
||||||
"name":"Radiant damage", "group": "Weakness/Resistance"
|
"name":"Radiant damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
"slashingMultiplier":{
|
"slashingMultiplier":{
|
||||||
"name":"Slashing damage", "group": "Weakness/Resistance"
|
"name":"Slashing damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
"thunderMultiplier":{
|
"thunderMultiplier":{
|
||||||
"name":"Thunder damage", "group": "Weakness/Resistance"
|
"name":"Thunder damage", "group": "Weakness/Resistance",
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -110,8 +111,8 @@ var operations = {
|
|||||||
Template.effectView.helpers({
|
Template.effectView.helpers({
|
||||||
sourceName: function(){
|
sourceName: function(){
|
||||||
var id = this.parent.id;
|
var id = this.parent.id;
|
||||||
if(!id) return;
|
if (!id) return;
|
||||||
switch(this.parent.collection){
|
switch (this.parent.collection){
|
||||||
case "Features":
|
case "Features":
|
||||||
return "Feature - " + Features.findOne(id, {fields: {name: 1}}).name;
|
return "Feature - " + Features.findOne(id, {fields: {name: 1}}).name;
|
||||||
case "Classes":
|
case "Classes":
|
||||||
@@ -130,33 +131,39 @@ Template.effectView.helpers({
|
|||||||
return stats[this.stat] && stats[this.stat].name || "No Stat";
|
return stats[this.stat] && stats[this.stat].name || "No Stat";
|
||||||
},
|
},
|
||||||
operationName: function(){
|
operationName: function(){
|
||||||
if(this.operation === "proficiency" ||
|
if (this.operation === "proficiency" ||
|
||||||
this.operation === "conditional") return null;
|
this.operation === "conditional") return null;
|
||||||
if(stats[this.stat].group === "Weakness/Resistance") return null;
|
if (stats[this.stat] && stats[this.stat].group === "Weakness/Resistance")
|
||||||
if(this.operation === "add" && evaluateEffect(this.charId, this) < 0) return null;
|
return null;
|
||||||
return operations[this.operation] && operations[this.operation].name || "No Operation";
|
if (this.operation === "add" && evaluateEffect(this.charId, this) < 0)
|
||||||
|
return null;
|
||||||
|
return operations[this.operation] &&
|
||||||
|
operations[this.operation].name || "No Operation";
|
||||||
},
|
},
|
||||||
statValue: function(){
|
statValue: function(){
|
||||||
if(this.operation === "advantage" ||
|
if (this.operation === "advantage" ||
|
||||||
this.operation === "disadvantage" ||
|
this.operation === "disadvantage" ||
|
||||||
this.operation === "fail"){
|
this.operation === "fail"){
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
if(this.operation === "proficiency"){
|
if (this.operation === "proficiency"){
|
||||||
if(this.value == 0.5 || this.calculation == 0.5) return "Half Proficiency";
|
if (this.value == 0.5 || this.calculation == 0.5)
|
||||||
if(this.value == 1 || this.calculation == 1) return "Proficiency";
|
return "Half Proficiency";
|
||||||
if(this.value == 2 || this.calculation == 2) return "Double Proficiency";
|
if (this.value == 1 || this.calculation == 1)
|
||||||
|
return "Proficiency";
|
||||||
|
if (this.value == 2 || this.calculation == 2)
|
||||||
|
return "Double Proficiency";
|
||||||
}
|
}
|
||||||
if(this.operation === "conditional"){
|
if (this.operation === "conditional"){
|
||||||
return this.calculation || this.value;
|
return this.calculation || this.value;
|
||||||
}
|
}
|
||||||
if(stats[this.stat].group === "Weakness/Resistance"){
|
if (stats[this.stat] && stats[this.stat].group === "Weakness/Resistance"){
|
||||||
if(this.value === 0.5) return "Resistance";
|
if (this.value === 0.5) return "Resistance";
|
||||||
if(this.value === 2) return "Vulnerability";
|
if (this.value === 2) return "Vulnerability";
|
||||||
if(this.value === 0) return "Immunity";
|
if (this.value === 0) return "Immunity";
|
||||||
}
|
}
|
||||||
var value = evaluateEffect(this.charId, this);
|
var value = evaluateEffect(this.charId, this);
|
||||||
if(_.isNumber(value)) return value;
|
if (_.isNumber(value)) return value;
|
||||||
return this.calculation || this.value;
|
return this.calculation || this.value;
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -32,7 +32,7 @@
|
|||||||
{{/if}}
|
{{/if}}
|
||||||
|
|
||||||
{{#if description}}
|
{{#if description}}
|
||||||
<div class="pre-wrap">{{evaluateString charId description}}</div>
|
<div>{{#markdown}}{{evaluateString charId description}}{{/markdown}}</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
|
|
||||||
{{> effectsViewList charId=charId parentId=_id}}
|
{{> effectsViewList charId=charId parentId=_id}}
|
||||||
|
|||||||
@@ -12,8 +12,9 @@
|
|||||||
{{>resource name="sorceryPoints" title="Sorcery Points" color="teal" char=this}}
|
{{>resource name="sorceryPoints" title="Sorcery Points" color="teal" char=this}}
|
||||||
<!--superiorityDice-->
|
<!--superiorityDice-->
|
||||||
{{>resource name="superiorityDice" title="Superiority Dice" color="teal" char=this}}
|
{{>resource name="superiorityDice" title="Superiority Dice" color="teal" char=this}}
|
||||||
|
|
||||||
<!--Attacks-->
|
<!--Attacks-->
|
||||||
|
<div>
|
||||||
<paper-shadow class="card">
|
<paper-shadow class="card">
|
||||||
<div class="top white">
|
<div class="top white">
|
||||||
Attacks
|
Attacks
|
||||||
@@ -48,8 +49,10 @@
|
|||||||
{{/each}}
|
{{/each}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
|
|
||||||
<!--Proficiencies-->
|
<!--Proficiencies-->
|
||||||
|
<div>
|
||||||
<paper-shadow class="card">
|
<paper-shadow class="card">
|
||||||
<div class="white top">
|
<div class="white top">
|
||||||
Proficiencies
|
Proficiencies
|
||||||
@@ -75,13 +78,15 @@
|
|||||||
{{/each}}
|
{{/each}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
|
|
||||||
<!--features-->
|
<!--features-->
|
||||||
{{#each features}}
|
{{#each features}}
|
||||||
|
<div>
|
||||||
<paper-shadow class="card featureCard"
|
<paper-shadow class="card featureCard"
|
||||||
hero-id="main" {{detailHero}}>
|
hero-id="main" {{detailHero}}>
|
||||||
<div class="top {{colorClass}} subhead"
|
<div class="top {{colorClass}} subhead"
|
||||||
layout horizontal
|
layout horizontal
|
||||||
hero-id="toolbar" {{detailHero}}>
|
hero-id="toolbar" {{detailHero}}>
|
||||||
<div flex hero-id="title" {{detailHero}}>
|
<div flex hero-id="title" {{detailHero}}>
|
||||||
{{name}}
|
{{name}}
|
||||||
@@ -94,57 +99,61 @@
|
|||||||
{{#if canEnable}}
|
{{#if canEnable}}
|
||||||
<core-tooltip label="Feature enabled"
|
<core-tooltip label="Feature enabled"
|
||||||
position="left">
|
position="left">
|
||||||
<paper-checkbox class="enabledCheckbox"
|
<paper-checkbox class="enabledCheckbox"
|
||||||
checked={{enabled}}>
|
checked={{enabled}}
|
||||||
|
disabled={{#unless canEditCharacter charId}}true{{/unless}}>
|
||||||
</paper-checkbox>
|
</paper-checkbox>
|
||||||
</core-tooltip>
|
</core-tooltip>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
{{#if description}}
|
{{#if description}}
|
||||||
<div flex class="bottom text"
|
<div flex class="bottom">
|
||||||
>{{evaluateString charId description}}</div>
|
{{#markdown}}{{evaluateString charId shortDescription}}{{/markdown}}
|
||||||
|
</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
{{#if hasUses}}
|
{{#if hasUses}}
|
||||||
<div layout horizontal center end-justified>
|
<div layout horizontal center end-justified>
|
||||||
<paper-button class="useFeature"
|
<paper-button class="useFeature"
|
||||||
disabled={{noUsesLeft}}>
|
disabled={{noUsesLeft}}>
|
||||||
Use
|
Use
|
||||||
</paper-button>
|
</paper-button>
|
||||||
<paper-button class="resetFeature"
|
<paper-button class="resetFeature"
|
||||||
disabled={{usesFull}}>
|
disabled={{usesFull}}>
|
||||||
Reset
|
Reset
|
||||||
</paper-button>
|
</paper-button>
|
||||||
</div>
|
</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
{{/each}}
|
{{/each}}
|
||||||
</div>
|
</div>
|
||||||
<div class="fab-buffer"></div>
|
<div class="fab-buffer"></div>
|
||||||
</div>
|
</div>
|
||||||
{{#if canEditCharacter _id}}
|
{{#if canEditCharacter _id}}
|
||||||
<paper-fab id="addFeature"
|
<paper-fab id="addFeature"
|
||||||
class="floatyButton"
|
class="floatyButton"
|
||||||
icon="add"
|
icon="add"
|
||||||
title="Add"
|
title="Add"
|
||||||
role="button"
|
role="button"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
aria-label="Add"
|
aria-label="Add"
|
||||||
hero-id="main"></paper-fab>
|
hero-id="main"></paper-fab>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template name="resource">
|
<template name="resource">
|
||||||
{{#if char.attributeBase name}}
|
{{#if characterCalculate "attributeBase" char._id name}}
|
||||||
<paper-shadow class="card"
|
<div>
|
||||||
|
<paper-shadow class="card"
|
||||||
hero-id="main" {{detailHero name char._id}}
|
hero-id="main" {{detailHero name char._id}}
|
||||||
layout horizontal>
|
layout horizontal>
|
||||||
<div class="left {{getColor}} display1 white-text"
|
<div class="left {{getColor}} display1 white-text"
|
||||||
hero-id="toolbar" {{detailHero name char._id}}
|
hero-id="toolbar" {{detailHero name char._id}}
|
||||||
layout horizontal center>
|
layout horizontal center>
|
||||||
<div style="margin-right: 8px;">
|
<div style="margin-right: 8px;">
|
||||||
<paper-icon-button class="resourceUp"
|
<paper-icon-button class="resourceUp"
|
||||||
icon="arrow-drop-up"
|
icon="arrow-drop-up"
|
||||||
disabled={{cantIncrement}}>
|
disabled={{cantIncrement}}>
|
||||||
</paper-icon-button>
|
</paper-icon-button>
|
||||||
<paper-icon-button class="resourceDown"
|
<paper-icon-button class="resourceDown"
|
||||||
@@ -152,13 +161,14 @@
|
|||||||
disabled={{cantDecrement}}>
|
disabled={{cantDecrement}}>
|
||||||
</paper-icon-button>
|
</paper-icon-button>
|
||||||
</div>
|
</div>
|
||||||
<div>{{char.attributeValue name}}</div>
|
<div>{{characterCalculate "attributeValue" char._id name}}</div>
|
||||||
<!--<div>/{{char.attributeBase name}}</div>-->
|
<!--<div>/{{char.attributeBase name}}</div>-->
|
||||||
</div>
|
</div>
|
||||||
<div class="right clickable"
|
<div class="right clickable"
|
||||||
flex layout horizontal center>
|
flex layout horizontal center>
|
||||||
{{title}}
|
{{title}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -3,14 +3,19 @@ Template.features.helpers({
|
|||||||
var features = Features.find({charId: this._id}, {sort: {color: 1, name: 1}});
|
var features = Features.find({charId: this._id}, {sort: {color: 1, name: 1}});
|
||||||
return features;
|
return features;
|
||||||
},
|
},
|
||||||
|
shortDescription: function() {
|
||||||
|
if (_.isString(this.description)){
|
||||||
|
return this.description.split(/^( *[-*_]){3,} *(?:\n+|$)/m)[0];
|
||||||
|
}
|
||||||
|
},
|
||||||
hasUses: function(){
|
hasUses: function(){
|
||||||
return this.usesValue() > 0;
|
return this.usesValue() > 0;
|
||||||
},
|
},
|
||||||
noUsesLeft: function(){
|
noUsesLeft: function(){
|
||||||
return this.usesLeft() <= 0;
|
return this.usesLeft() <= 0 || !canEditCharacter(this.charId);
|
||||||
},
|
},
|
||||||
usesFull: function(){
|
usesFull: function(){
|
||||||
return this.usesLeft() >= this.usesValue();
|
return this.usesLeft() >= this.usesValue() || !canEditCharacter(this.charId);
|
||||||
},
|
},
|
||||||
colorClass: function(){
|
colorClass: function(){
|
||||||
return getColorClass(this.color);
|
return getColorClass(this.color);
|
||||||
@@ -96,16 +101,19 @@ Template.features.events({
|
|||||||
|
|
||||||
Template.resource.helpers({
|
Template.resource.helpers({
|
||||||
cantIncrement: function(){
|
cantIncrement: function(){
|
||||||
var baseBigger = this.char.attributeValue(this.name) <
|
var value = Characters.calculate.attributeValue(this.char._id, this.name);
|
||||||
this.char.attributeBase(this.name);
|
var base = Characters.calculate.attributeBase(this.char._id, this.name);
|
||||||
return !baseBigger;
|
var baseBigger = value < base;
|
||||||
|
return !baseBigger || !canEditCharacter(this.char._id);
|
||||||
},
|
},
|
||||||
cantDecrement: function(){
|
cantDecrement: function(){
|
||||||
var valuePositive = this.char.attributeValue(this.name) > 0;
|
var value = Characters.calculate.attributeValue(this.char._id, this.name);
|
||||||
return !valuePositive;
|
var valuePositive = value > 0;
|
||||||
|
return !valuePositive || !canEditCharacter(this.char._id);
|
||||||
},
|
},
|
||||||
getColor: function(){
|
getColor: function(){
|
||||||
if (this.char.attributeValue(this.name) > 0){
|
var value = Characters.calculate.attributeValue(this.char._id, this.name);
|
||||||
|
if (value > 0){
|
||||||
return this.color;
|
return this.color;
|
||||||
} else {
|
} else {
|
||||||
return "grey";
|
return "grey";
|
||||||
@@ -115,14 +123,17 @@ Template.resource.helpers({
|
|||||||
|
|
||||||
Template.resource.events({
|
Template.resource.events({
|
||||||
"tap .resourceUp": function(event){
|
"tap .resourceUp": function(event){
|
||||||
if (this.char.attributeValue(this.name) < this.char.attributeBase(this.name)){
|
var value = Characters.calculate.attributeValue(this.char._id, this.name);
|
||||||
|
var base = Characters.calculate.attributeBase(this.char._id, this.name);
|
||||||
|
if (value < base){
|
||||||
var modifier = {$inc: {}};
|
var modifier = {$inc: {}};
|
||||||
modifier.$inc[this.name + ".adjustment"] = 1;
|
modifier.$inc[this.name + ".adjustment"] = 1;
|
||||||
Characters.update(this.char._id, modifier, {validate: false});
|
Characters.update(this.char._id, modifier, {validate: false});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"tap .resourceDown": function(event){
|
"tap .resourceDown": function(event){
|
||||||
if (this.char.attributeValue(this.name) > 0){
|
var value = Characters.calculate.attributeValue(this.char._id, this.name);
|
||||||
|
if (value > 0){
|
||||||
var modifier = {$inc: {}};
|
var modifier = {$inc: {}};
|
||||||
modifier.$inc[this.name + ".adjustment"] = -1;
|
modifier.$inc[this.name + ".adjustment"] = -1;
|
||||||
Characters.update(this.char._id, modifier, {validate: false});
|
Characters.update(this.char._id, modifier, {validate: false});
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
<template name="carryCapacityBar">
|
||||||
|
<div class="carryCapacityBar">
|
||||||
|
<div class="carriedWeightBar"
|
||||||
|
style="width: {{carriedPercent}}%;
|
||||||
|
background-color: {{carriedColor}}">
|
||||||
|
</div>
|
||||||
|
<div class="tick"
|
||||||
|
style="width: 33.333%;">
|
||||||
|
</div>
|
||||||
|
<div class="tick"
|
||||||
|
style="width: 66.666%;">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{#if overCarriedPercent}}
|
||||||
|
<div class="carryCapacityBar">
|
||||||
|
<div class="carriedWeightBar"
|
||||||
|
style="width: {{overCarriedPercent}}%;
|
||||||
|
background-color: {{overCarriedColor}}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{/if}}
|
||||||
|
</template>
|
||||||
@@ -0,0 +1,67 @@
|
|||||||
|
var getFractionCarried = function(char) {
|
||||||
|
//find out the weight
|
||||||
|
var weight = 0;
|
||||||
|
Containers.find(
|
||||||
|
{charId: char._id, isCarried: true}
|
||||||
|
).forEach(function(container){
|
||||||
|
weight += container.totalWeight();
|
||||||
|
});
|
||||||
|
Items.find(
|
||||||
|
{charId: char._id, "parent.id": char._id},
|
||||||
|
{fields: {weight : 1, quantity: 1}}
|
||||||
|
).forEach(function(item){
|
||||||
|
weight += item.totalWeight();
|
||||||
|
});
|
||||||
|
//get strength
|
||||||
|
var strength = Characters.calculate.attributeValue(char._id, "strength");
|
||||||
|
var carryMultiplier = Characters.calculate
|
||||||
|
.attributeValue(char._id, "carryMultiplier");
|
||||||
|
var capacity = strength * 15 * carryMultiplier;
|
||||||
|
return weight / capacity;
|
||||||
|
};
|
||||||
|
|
||||||
|
Template.carryCapacityBar.onCreated(function() {
|
||||||
|
var self = this;
|
||||||
|
self.carriedFraction = new ReactiveVar(0);
|
||||||
|
self.autorun(function() {
|
||||||
|
self.carriedFraction.set(getFractionCarried(self.data));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Template.carryCapacityBar.helpers({
|
||||||
|
carriedPercent: function() {
|
||||||
|
var percent = 100 * Template.instance().carriedFraction.get();
|
||||||
|
return percent > 100 ? 100 : percent;
|
||||||
|
},
|
||||||
|
overCarriedPercent: function() {
|
||||||
|
var percent = 100 * Template.instance().carriedFraction.get();
|
||||||
|
var overPercent = percent - 100;
|
||||||
|
if (overPercent < 0) return 0;
|
||||||
|
if (overPercent > 100) return 100;
|
||||||
|
return overPercent;
|
||||||
|
},
|
||||||
|
carriedColor: function() {
|
||||||
|
var frac = Template.instance().carriedFraction.get();
|
||||||
|
if (frac < 1 / 3){
|
||||||
|
return "#2196F3";
|
||||||
|
} else if (frac < 2 / 3){
|
||||||
|
return "#CDDC39";
|
||||||
|
} else if (frac < 1) {
|
||||||
|
return "#FFC107";
|
||||||
|
} else {
|
||||||
|
return "#F44336";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
overCarriedColor: function() {
|
||||||
|
var frac = Template.instance().carriedFraction.get();
|
||||||
|
if (frac < 1 / 3){
|
||||||
|
return "#2196F3";
|
||||||
|
} else if (frac < 2 / 3){
|
||||||
|
return "#CDDC39";
|
||||||
|
} else if (frac < 1) {
|
||||||
|
return "#FFC107";
|
||||||
|
} else {
|
||||||
|
return "#F44336";
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
.carryCapacityBar {
|
||||||
|
background-color: #7DC580;
|
||||||
|
background-color: rgba(255,255,255,0.27);
|
||||||
|
position: relative;
|
||||||
|
height: 4px;
|
||||||
|
div{
|
||||||
|
height: 100%;
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
.tick {
|
||||||
|
border-right: solid 2px #E5E5E5;
|
||||||
|
border-right-color: rgba(255,255,255,0.54);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
<template name="carryDialog">
|
||||||
|
{{#baseDialog title="Weight Carried" class=color hideEdit=true}}
|
||||||
|
<div layout horizontal center-justified end>
|
||||||
|
<div class="display2">
|
||||||
|
{{round carriedWeight 1}}
|
||||||
|
</div>
|
||||||
|
<div class="display1">
|
||||||
|
lbs
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr class="vertMargin">
|
||||||
|
|
||||||
|
{{> carryCapacityTable}}
|
||||||
|
|
||||||
|
{{/baseDialog}}
|
||||||
|
</template>
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
Template.carryDialog.helpers({
|
||||||
|
carriedWeight: function() {
|
||||||
|
var weight = 0;
|
||||||
|
Containers.find(
|
||||||
|
{charId: this.charId, isCarried: true}
|
||||||
|
).forEach(function(container){
|
||||||
|
weight += container.totalWeight();
|
||||||
|
});
|
||||||
|
Items.find(
|
||||||
|
{charId: this.charId, "parent.id": this.charId},
|
||||||
|
{fields: {weight : 1, quantity: 1}}
|
||||||
|
).forEach(function(item){
|
||||||
|
weight += item.totalWeight();
|
||||||
|
});
|
||||||
|
return weight;
|
||||||
|
},
|
||||||
|
color: function() {
|
||||||
|
if (this.color) return this.color + " white-text";
|
||||||
|
},
|
||||||
|
});
|
||||||
@@ -34,13 +34,13 @@
|
|||||||
<template name="containerView">
|
<template name="containerView">
|
||||||
<div layout horizontal wrap center justified>
|
<div layout horizontal wrap center justified>
|
||||||
<table class="summaryTable fullwidth">
|
<table class="summaryTable fullwidth">
|
||||||
<tr><td>Container</td><td>{{weight}}lbs</td><td>{{longValueString value}}</td></tr>
|
<tr><td>Container</td><td>{{round weight}}lbs</td><td>{{longValueString value}}</td></tr>
|
||||||
<tr><td>Contents</td><td>{{contentsWeight}}lbs</td><td>{{longValueString contentsValue}}</td></tr>
|
<tr><td>Contents</td><td>{{round contentsWeight}}lbs</td><td>{{longValueString contentsValue}}</td></tr>
|
||||||
<tr class="body2"><td>Total</td><td>{{totalWeight}}lbs</td><td>{{longValueString totalValue}}</td></tr>
|
<tr class="body2"><td>Total</td><td>{{round totalWeight}}lbs</td><td>{{longValueString totalValue}}</td></tr>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
{{#if description}}
|
{{#if description}}
|
||||||
<hr class="vertMargin">
|
<hr class="vertMargin">
|
||||||
<div class="pre-wrap">{{evaluateString charId description}}</div>
|
<div>{{#markdown}}{{evaluateString charId description}}{{/markdown}}</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -3,24 +3,57 @@
|
|||||||
<div id="inventory" class="scroll-y" fit>
|
<div id="inventory" class="scroll-y" fit>
|
||||||
<div class="column-container">
|
<div class="column-container">
|
||||||
<!--Net Worth-->
|
<!--Net Worth-->
|
||||||
<paper-shadow class="card" layout horizontal>
|
<div>
|
||||||
<div class="indigo white-text subhead left">
|
<paper-shadow class="card">
|
||||||
Net Worth
|
<div class="white top" layout horizontal center>
|
||||||
</div>
|
<div class="subhead" flex>
|
||||||
<div class="right" flex>
|
Net Worth
|
||||||
{{valueString netWorth}}
|
</div>
|
||||||
|
<div>
|
||||||
|
{{valueString netWorth}}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
<!--Weight Carried-->
|
<!--Weight Carried-->
|
||||||
<paper-shadow class="card" layout horizontal>
|
<div>
|
||||||
<div class="green white-text subhead left">
|
<paper-shadow class="card"
|
||||||
Weight Carried
|
hero-id="main" {{detailHero "weightCarried" _id}}>
|
||||||
|
<div class="top green white-text weightCarried"
|
||||||
|
hero-id="toolbar" {{detailHero "weightCarried" _id}}
|
||||||
|
layout horizontal center>
|
||||||
|
<div class="subhead" flex>
|
||||||
|
Weight Carried
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
{{round weightCarried}}lbs
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="right" flex>
|
<div class="bottom green" style="padding: 0;">
|
||||||
{{round weightCarried}}lbs
|
{{> carryCapacityBar}}
|
||||||
</div>
|
</div>
|
||||||
|
{{#if encumberedBuffs.count}}
|
||||||
|
<div class="bottom list">
|
||||||
|
{{#each encumberedBuffs}}
|
||||||
|
<div class="item-slot">
|
||||||
|
<div class="item buff"
|
||||||
|
hero-id="main" {{detailHero}}
|
||||||
|
layout horizontal center>
|
||||||
|
<div flex>
|
||||||
|
<core-icon icon="work"
|
||||||
|
style="margin-right: 16px">
|
||||||
|
</core-icon>
|
||||||
|
{{name}}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{/each}}
|
||||||
|
</div>
|
||||||
|
{{/if}}
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
<!--Equipment-->
|
<!--Equipment-->
|
||||||
|
<div>
|
||||||
<paper-shadow class="card equipmentContainer">
|
<paper-shadow class="card equipmentContainer">
|
||||||
<div class="white top" layout horizontal center>
|
<div class="white top" layout horizontal center>
|
||||||
<div class="subhead" flex>
|
<div class="subhead" flex>
|
||||||
@@ -48,10 +81,12 @@
|
|||||||
{{/each}}
|
{{/each}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
<!--Carried Items-->
|
<!--Carried Items-->
|
||||||
|
<div>
|
||||||
<paper-shadow class="card carriedContainer">
|
<paper-shadow class="card carriedContainer">
|
||||||
<div class="white top" layout horizontal center>
|
<div class="white top" layout horizontal center>
|
||||||
<div class="subhead">
|
<div class="subhead" flex>
|
||||||
Carried
|
Carried
|
||||||
</div>
|
</div>
|
||||||
<div class="caption" style="margin-right: 8px">
|
<div class="caption" style="margin-right: 8px">
|
||||||
@@ -67,8 +102,10 @@
|
|||||||
{{/each}}
|
{{/each}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
{{#each containers}}
|
{{#each containers}}
|
||||||
<paper-shadow class="card itemContainer"
|
<div>
|
||||||
|
<paper-shadow class="card itemContainer"
|
||||||
hero-id="main" {{detailHero}}>
|
hero-id="main" {{detailHero}}>
|
||||||
<div class="top {{colorClass}}"
|
<div class="top {{colorClass}}"
|
||||||
hero-id="toolbar" {{detailHero}}
|
hero-id="toolbar" {{detailHero}}
|
||||||
@@ -85,6 +122,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<core-tooltip label="Container carried" position="left">
|
<core-tooltip label="Container carried" position="left">
|
||||||
<paper-checkbox class="carriedCheckbox"
|
<paper-checkbox class="carriedCheckbox"
|
||||||
|
disabled={{#unless canEditCharacter charId}}true{{/unless}}
|
||||||
checked={{isCarried}}>
|
checked={{isCarried}}>
|
||||||
</paper-checkbox>
|
</paper-checkbox>
|
||||||
</core-tooltip>
|
</core-tooltip>
|
||||||
@@ -95,6 +133,7 @@
|
|||||||
{{/each}}
|
{{/each}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
{{/each}}
|
{{/each}}
|
||||||
</div>
|
</div>
|
||||||
<div class="fab-buffer"></div>
|
<div class="fab-buffer"></div>
|
||||||
@@ -123,11 +162,11 @@
|
|||||||
<div class="item {{hidden}} inventoryItem"
|
<div class="item {{hidden}} inventoryItem"
|
||||||
hero-id="main" {{detailHero}}
|
hero-id="main" {{detailHero}}
|
||||||
layout horizontal center
|
layout horizontal center
|
||||||
draggable="true">
|
draggable={{canEditCharacter charId}}>
|
||||||
<div flex class="itemName">
|
<div flex class="itemName">
|
||||||
{{#if ne1 quantity}}{{quantity}} {{/if}}{{pluralName}}
|
{{#if ne1 quantity}}{{quantity}} {{/if}}{{pluralName}}
|
||||||
</div>
|
</div>
|
||||||
{{#if settings.showIncrement}}
|
{{#if settings.showIncrement}}{{#if canEditCharacter charId}}
|
||||||
<div class="incrementButtons">
|
<div class="incrementButtons">
|
||||||
<paper-icon-button class="addItemQuantity"
|
<paper-icon-button class="addItemQuantity"
|
||||||
icon="add"
|
icon="add"
|
||||||
@@ -137,7 +176,7 @@
|
|||||||
icon="remove"
|
icon="remove"
|
||||||
style="margin-right: -8px"></paper-icon-button>
|
style="margin-right: -8px"></paper-icon-button>
|
||||||
</div>
|
</div>
|
||||||
{{/if}}
|
{{/if}}{{/if}}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -44,6 +44,12 @@ Template.inventory.helpers({
|
|||||||
).forEach(function(item){
|
).forEach(function(item){
|
||||||
worth += item.totalValue();
|
worth += item.totalValue();
|
||||||
});
|
});
|
||||||
|
Containers.find(
|
||||||
|
{charId: this._id},
|
||||||
|
{fields: {value : 1}}
|
||||||
|
).forEach(function(container) {
|
||||||
|
if (container.value) worth += container.value;
|
||||||
|
});
|
||||||
return worth;
|
return worth;
|
||||||
},
|
},
|
||||||
weightCarried: function(){
|
weightCarried: function(){
|
||||||
@@ -61,6 +67,18 @@ Template.inventory.helpers({
|
|||||||
});
|
});
|
||||||
return weight;
|
return weight;
|
||||||
},
|
},
|
||||||
|
encumberedBuffs: function(){
|
||||||
|
return Buffs.find({
|
||||||
|
charId: this._id,
|
||||||
|
type: "inate",
|
||||||
|
name: {$in: [
|
||||||
|
"Encumbered",
|
||||||
|
"Heavily encumbered",
|
||||||
|
"Over encumbered",
|
||||||
|
"Can't move load",
|
||||||
|
]},
|
||||||
|
});
|
||||||
|
},
|
||||||
equipmentValue: function(){
|
equipmentValue: function(){
|
||||||
var value = 0;
|
var value = 0;
|
||||||
Items.find(
|
Items.find(
|
||||||
@@ -136,6 +154,23 @@ Template.inventory.events({
|
|||||||
heroId: containerId,
|
heroId: containerId,
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
"tap .weightCarried": function(event) {
|
||||||
|
var charId = this._id;
|
||||||
|
GlobalUI.setDetail({
|
||||||
|
template: "carryDialog",
|
||||||
|
data: {charId: charId, color: "green"},
|
||||||
|
heroId: charId + "weightCarried",
|
||||||
|
});
|
||||||
|
},
|
||||||
|
"tap .buff": function(event){
|
||||||
|
var buffId = this._id;
|
||||||
|
var charId = Template.parentData()._id;
|
||||||
|
GlobalUI.setDetail({
|
||||||
|
template: "buffDialog",
|
||||||
|
data: {buffId: buffId, charId: charId},
|
||||||
|
heroId: buffId,
|
||||||
|
});
|
||||||
|
},
|
||||||
"tap .inventoryItem": function(event){
|
"tap .inventoryItem": function(event){
|
||||||
var itemId = this._id;
|
var itemId = this._id;
|
||||||
var charId = Template.parentData()._id;
|
var charId = Template.parentData()._id;
|
||||||
@@ -145,6 +180,20 @@ Template.inventory.events({
|
|||||||
heroId: itemId,
|
heroId: itemId,
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
"hold .inventoryItem": function(event, instance) {
|
||||||
|
var itemId = this._id;
|
||||||
|
var charId = Template.parentData()._id;
|
||||||
|
var containerId = this.parent.id;
|
||||||
|
GlobalUI.showDialog({
|
||||||
|
template: "moveItemDialog",
|
||||||
|
data: {
|
||||||
|
charId: charId,
|
||||||
|
itemId: itemId,
|
||||||
|
containerId: containerId,
|
||||||
|
},
|
||||||
|
heading: "Move " + this.pluralName(),
|
||||||
|
});
|
||||||
|
},
|
||||||
"tap .incrementButtons": function(event) {
|
"tap .incrementButtons": function(event) {
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -18,8 +18,8 @@
|
|||||||
{{#if requiresAttunement}}<div class="vertMargin">Requires Attunement</div>{{/if}}
|
{{#if requiresAttunement}}<div class="vertMargin">Requires Attunement</div>{{/if}}
|
||||||
</div>
|
</div>
|
||||||
{{#if description}}
|
{{#if description}}
|
||||||
<hr class="vertMargin">
|
<hr style="margin: 16px 0 16px 0;">
|
||||||
<div class="pre-wrap">{{evaluateString charId description}}</div>
|
<div>{{#markdown}}{{evaluateString charId description}}{{/markdown}}</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
{{> effectsViewList charId=charId parentId=_id}}
|
{{> effectsViewList charId=charId parentId=_id}}
|
||||||
{{> attacksViewList charId=charId parentId=_id}}
|
{{> attacksViewList charId=charId parentId=_id}}
|
||||||
|
|||||||
@@ -0,0 +1,7 @@
|
|||||||
|
html /deep/ .moveItemDialog paper-tabs::shadow #selectionBar {
|
||||||
|
background-color: #D50000;
|
||||||
|
}
|
||||||
|
|
||||||
|
html /deep/ .moveItemDialog paper-tab::shadow #ink {
|
||||||
|
color: #D50000;
|
||||||
|
}
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
<template name="moveItemDialog">
|
||||||
|
<div class="moveItemDialog">
|
||||||
|
<paper-tabs selected="{{selectedTab}}">
|
||||||
|
<paper-tab name="containers"
|
||||||
|
class="clickable">
|
||||||
|
Containers
|
||||||
|
</paper-tab>
|
||||||
|
<paper-tab name="characters"
|
||||||
|
class="clickable">
|
||||||
|
Characters
|
||||||
|
</paper-tab>
|
||||||
|
</paper-tabs>
|
||||||
|
<core-animated-pages selected="{{selectedTab}}"
|
||||||
|
transitions="slide-from-right"
|
||||||
|
style="width: 250px;
|
||||||
|
height: 200px;
|
||||||
|
overflow-y: auto;
|
||||||
|
overflow-x: hidden;">
|
||||||
|
<section name="containers">
|
||||||
|
<core-menu id="containerMenu" style="margin: 0;">
|
||||||
|
{{#each containers}}
|
||||||
|
<paper-item name={{_id}}
|
||||||
|
layout horizontal center>
|
||||||
|
<core-icon icon="image:brightness-1"
|
||||||
|
style="color: {{hexColor color}};
|
||||||
|
margin-right: 16px;">
|
||||||
|
</core-icon>
|
||||||
|
<div>{{name}}</div>
|
||||||
|
</paper-item>
|
||||||
|
{{/each}}
|
||||||
|
</core-menu>
|
||||||
|
</section>
|
||||||
|
<section name="characters">
|
||||||
|
<core-menu id="characterMenu" style="margin: 0;">
|
||||||
|
{{#each characters}}
|
||||||
|
<paper-item name={{_id}}
|
||||||
|
layout horizontal center>
|
||||||
|
<div class="item small">
|
||||||
|
{{name}}
|
||||||
|
</div>
|
||||||
|
</paper-item>
|
||||||
|
{{/each}}
|
||||||
|
</core-menu>
|
||||||
|
</section>
|
||||||
|
</core-animated-pages>
|
||||||
|
</div>
|
||||||
|
<paper-button id="cancelButton" affirmative> Cancel </paper-button>
|
||||||
|
<paper-button id="moveButton" affirmative> Move </paper-button>
|
||||||
|
</template>
|
||||||
@@ -0,0 +1,56 @@
|
|||||||
|
Template.moveItemDialog.onCreated(function() {
|
||||||
|
Session.setDefault("moveItemDialogTab", "containers");
|
||||||
|
});
|
||||||
|
|
||||||
|
Template.moveItemDialog.helpers({
|
||||||
|
selectedTab: function() {
|
||||||
|
return Session.get("moveItemDialogTab");
|
||||||
|
},
|
||||||
|
characters: function() {
|
||||||
|
var userId = Meteor.userId();
|
||||||
|
return Characters.find(
|
||||||
|
{
|
||||||
|
$or: [
|
||||||
|
{readers: userId},
|
||||||
|
{writers: userId},
|
||||||
|
{owner: userId},
|
||||||
|
],
|
||||||
|
_id: {$ne: this.charId},
|
||||||
|
},
|
||||||
|
{fields: {name: 1}}
|
||||||
|
);
|
||||||
|
},
|
||||||
|
containers: function(){
|
||||||
|
return Containers.find(
|
||||||
|
{
|
||||||
|
charId: this.charId,
|
||||||
|
_id: {$ne: this.containerId},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
fields: {color: 1, name: 1},
|
||||||
|
sort: {color: 1, name: 1},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
Template.moveItemDialog.events({
|
||||||
|
"tap paper-tab": function(event) {
|
||||||
|
Session.set("moveItemDialogTab", event.currentTarget.getAttribute("name"));
|
||||||
|
},
|
||||||
|
"tap #moveButton": function(event, instance) {
|
||||||
|
var tab = Session.get("moveItemDialogTab");
|
||||||
|
if (tab === "containers"){
|
||||||
|
var containerId = instance.find("#containerMenu").selected;
|
||||||
|
if (!containerId) throw "no menu selection";
|
||||||
|
Meteor.call("moveItemToContainer", this.itemId, containerId);
|
||||||
|
} else if (tab === "characters"){
|
||||||
|
var characterId = instance.find("#characterMenu").selected;
|
||||||
|
if (!characterId) throw "no menu selection";
|
||||||
|
Meteor.call("moveItemToCharacter", this.itemId, characterId);
|
||||||
|
} else {
|
||||||
|
throw "Move item dialog tab is not set to containers or character," +
|
||||||
|
" it is set to " + tab;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
@@ -6,23 +6,27 @@
|
|||||||
</div>
|
</div>
|
||||||
{{#if description}}
|
{{#if description}}
|
||||||
<hr class="vertMargin">
|
<hr class="vertMargin">
|
||||||
<div class="pre-wrap">{{description}}</div>
|
<div>{{#markdown}}{{description}}{{/markdown}}</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
{{else}}
|
{{else}}
|
||||||
<div horizontal layout>
|
{{> experienceEdit}}
|
||||||
<!--Name-->
|
|
||||||
<paper-input id="experienceNameInput" label="Name" floatinglabel value={{name}} flex></paper-input>
|
|
||||||
<!--Value-->
|
|
||||||
<paper-input-decorator label="Value" floatinglabel>
|
|
||||||
<input id="valueInput" type="number" value={{value}}>
|
|
||||||
</paper-input-decorator>
|
|
||||||
</div>
|
|
||||||
<!--Description-->
|
|
||||||
<paper-input-decorator label="Description" floatinglabel layout vertical>
|
|
||||||
<paper-autogrow-textarea>
|
|
||||||
<textarea id="experienceDescriptionInput" placeholder value={{description}}></textarea>
|
|
||||||
</paper-autogrow-textarea>
|
|
||||||
</paper-input-decorator>
|
|
||||||
{{/baseDialog}}
|
{{/baseDialog}}
|
||||||
{{/with}}
|
{{/with}}
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
<template name="experienceEdit">
|
||||||
|
<div horizontal layout>
|
||||||
|
<!--Name-->
|
||||||
|
<paper-input id="experienceNameInput" label="Name" floatinglabel value={{name}} flex></paper-input>
|
||||||
|
<!--Value-->
|
||||||
|
<paper-input-decorator label="Value" floatinglabel>
|
||||||
|
<input id="valueInput" type="number" value={{value}}>
|
||||||
|
</paper-input-decorator>
|
||||||
|
</div>
|
||||||
|
<!--Description-->
|
||||||
|
<paper-input-decorator label="Description" floatinglabel layout vertical>
|
||||||
|
<paper-autogrow-textarea>
|
||||||
|
<textarea id="experienceDescriptionInput" placeholder value={{description}}></textarea>
|
||||||
|
</paper-autogrow-textarea>
|
||||||
|
</paper-input-decorator>
|
||||||
|
</template>
|
||||||
|
|||||||
@@ -1,3 +1,7 @@
|
|||||||
|
Template.experienceEdit.onRendered(function(){
|
||||||
|
updatePolymerInputs(this);
|
||||||
|
});
|
||||||
|
|
||||||
Template.experienceDialog.helpers({
|
Template.experienceDialog.helpers({
|
||||||
experience: function(){
|
experience: function(){
|
||||||
Experiences.findOne(this.experienceId);
|
Experiences.findOne(this.experienceId);
|
||||||
@@ -18,8 +22,10 @@ Template.experienceDialog.events({
|
|||||||
);
|
);
|
||||||
GlobalUI.closeDetail();
|
GlobalUI.closeDetail();
|
||||||
},
|
},
|
||||||
//TODO validate input (integer, non-negative, etc) for these inputs and give validation errors
|
});
|
||||||
"change #experienceNameInput, input #experienceNameInput": function(event){
|
|
||||||
|
Template.experienceEdit.events({
|
||||||
|
"change #experienceNameInput": function(event){
|
||||||
var value = event.currentTarget.value;
|
var value = event.currentTarget.value;
|
||||||
Experiences.update(this._id, {$set: {name: value}});
|
Experiences.update(this._id, {$set: {name: value}});
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -3,14 +3,15 @@
|
|||||||
<div id="journal" class="scroll-y" fit>
|
<div id="journal" class="scroll-y" fit>
|
||||||
<div class="column-container">
|
<div class="column-container">
|
||||||
<!--Experience Table-->
|
<!--Experience Table-->
|
||||||
<paper-shadow class="card experiencesCard"
|
<div><paper-shadow class="card experiencesCard"
|
||||||
hero-id="main" {{detailHero}}>
|
hero-id="main" {{detailHero}}>
|
||||||
<div class="top white subhead"
|
<div class="top white subhead"
|
||||||
hero-id="toolbar" {{detailHero}}
|
hero-id="toolbar" {{detailHero}}
|
||||||
layout horizontal center>
|
layout horizontal center>
|
||||||
<div flex>Experience</div>
|
<div flex>Experience</div>
|
||||||
<div >{{experience}} XP</div>
|
<div >{{characterCalculate "experience" _id}} XP</div>
|
||||||
<paper-icon-button class="black54" id="addXP" icon="add"></paper-icon-button>
|
<paper-icon-button class="black54" id="addXP" icon="add"
|
||||||
|
disabled={{#unless canEditCharacter _id}}true{{/unless}}></paper-icon-button>
|
||||||
</div>
|
</div>
|
||||||
<div class="bottom list">
|
<div class="bottom list">
|
||||||
{{#each experiences}}
|
{{#each experiences}}
|
||||||
@@ -36,16 +37,16 @@
|
|||||||
</paper-button>
|
</paper-button>
|
||||||
</div>
|
</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</paper-shadow>
|
</paper-shadow></div>
|
||||||
<!--Class Table-->
|
<!--Class Table-->
|
||||||
<paper-shadow class="card"
|
<div><paper-shadow class="card"
|
||||||
hero-id="main" {{detailHero}}>
|
hero-id="main" {{detailHero}}>
|
||||||
<div class="white top"
|
<div class="white top"
|
||||||
hero-id="toolbar" {{detailHero}}
|
hero-id="toolbar" {{detailHero}}
|
||||||
layout horizontal center>
|
layout horizontal center>
|
||||||
<div flex>
|
<div flex>
|
||||||
<div class="containerName subhead">
|
<div class="containerName subhead">
|
||||||
Level {{level}}
|
Level {{characterCalculate "level" _id}}
|
||||||
</div>
|
</div>
|
||||||
{{#if nextLevelXP}}
|
{{#if nextLevelXP}}
|
||||||
<div class="caption">
|
<div class="caption">
|
||||||
@@ -55,7 +56,8 @@
|
|||||||
</div>
|
</div>
|
||||||
<paper-icon-button class="black54"
|
<paper-icon-button class="black54"
|
||||||
id="addClassButton"
|
id="addClassButton"
|
||||||
icon="add">
|
icon="add"
|
||||||
|
disabled={{#unless canEditCharacter _id}}true{{/unless}}>
|
||||||
</paper-icon-button>
|
</paper-icon-button>
|
||||||
</div>
|
</div>
|
||||||
<div class="bottom list">
|
<div class="bottom list">
|
||||||
@@ -76,29 +78,31 @@
|
|||||||
</div>
|
</div>
|
||||||
{{/each}}
|
{{/each}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow></div>
|
||||||
<!--Notes-->
|
<!--Notes-->
|
||||||
{{#each notes}}
|
{{#each notes}}
|
||||||
|
<div>
|
||||||
<paper-shadow class="card" hero-id="main" {{detailHero}}>
|
<paper-shadow class="card" hero-id="main" {{detailHero}}>
|
||||||
<div class="top {{colorClass}} noteTop subhead"
|
<div class="top {{colorClass}} noteTop subhead"
|
||||||
hero-id="toolbar" {{detailHero}}
|
hero-id="toolbar" {{detailHero}}
|
||||||
layout horizontal center>
|
layout horizontal center>
|
||||||
{{name}}
|
{{name}}
|
||||||
</div>
|
</div>
|
||||||
<div class="bottom text">{{description}}</div>
|
<div class="bottom">{{#markdown}}{{description}}{{/markdown}}</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
{{/each}}
|
{{/each}}
|
||||||
</div>
|
</div>
|
||||||
<div class="fab-buffer"></div>
|
<div class="fab-buffer"></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{{#if canEditCharacter _id}}
|
{{#if canEditCharacter _id}}
|
||||||
<paper-fab id="addNote"
|
<paper-fab id="addNote"
|
||||||
class="floatyButton"
|
class="floatyButton"
|
||||||
icon="add"
|
icon="add"
|
||||||
title="Add"
|
title="Add"
|
||||||
role="button"
|
role="button"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
hero-id="main"></paper-fab>
|
hero-id="main"></paper-fab>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ Template.journal.helpers({
|
|||||||
return Levels.find({charId: charId, classId: this._id}, {sort: {value: 1}});
|
return Levels.find({charId: charId, classId: this._id}, {sort: {value: 1}});
|
||||||
},
|
},
|
||||||
nextLevelXP: function(){
|
nextLevelXP: function(){
|
||||||
var currentLevel = this.level();
|
var currentLevel = Characters.calculate.level(this._id);
|
||||||
if (currentLevel < 20){
|
if (currentLevel < 20){
|
||||||
return XP_TABLE[currentLevel];
|
return XP_TABLE[currentLevel];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<template name="noteDialog">
|
<template name="noteDialog">
|
||||||
{{#with note}}
|
{{#with note}}
|
||||||
{{#baseDialog title=name class=colorClass startEditing=../startEditing}}
|
{{#baseDialog title=name class=colorClass startEditing=../startEditing}}
|
||||||
<div class="pre-wrap">{{description}}</div>
|
<div>{{#markdown}}{{description}}{{/markdown}}</div>
|
||||||
{{else}}
|
{{else}}
|
||||||
{{> noteDialogEdit}}
|
{{> noteDialogEdit}}
|
||||||
{{/baseDialog}}
|
{{/baseDialog}}
|
||||||
|
|||||||
@@ -0,0 +1,9 @@
|
|||||||
|
<template name="backgroundDialog">
|
||||||
|
{{#baseDialog title=title class=colorClass hideColor="true" hideDelete="true"}}
|
||||||
|
<div>{{#markdown}}{{evaluateString charId value}}{{/markdown}}</div>
|
||||||
|
{{> proficiencyViewList charId=charId parentId=charId parentGroup="background"}}
|
||||||
|
{{else}}
|
||||||
|
{{> textDialogEdit}}
|
||||||
|
{{> proficiencyEditList parentId=charId parentCollection="Characters" charId=charId parentGroup="background"}}
|
||||||
|
{{/baseDialog}}
|
||||||
|
</template>
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
Template.backgroundDialog.helpers({
|
||||||
|
value: function(){
|
||||||
|
var fieldSelector = {fields: {}};
|
||||||
|
fieldSelector.fields[this.field] = 1;
|
||||||
|
var char = Characters.findOne(this.charId, fieldSelector);
|
||||||
|
return char[this.field];
|
||||||
|
}
|
||||||
|
});
|
||||||
@@ -1,18 +1,36 @@
|
|||||||
<template name="personaDetailsDialog">
|
<template name="personaDetailsDialog">
|
||||||
{{#baseDialog title=name class="deep-purple white-text" hideColor="true" hideDelete="true" startEditing=startEditing}}
|
{{#baseDialog title=name class="deep-purple white-text" hideColor="true" hideDelete="true" startEditing=startEditing}}
|
||||||
{{alignment}} {{gender}} {{race}}
|
{{#with char}}
|
||||||
|
<div>{{alignment}} {{gender}} {{race}}</div>
|
||||||
|
<core-image style="width: 350px; height: 350px; margin-top: 8px;"
|
||||||
|
sizing="cover"
|
||||||
|
hero-id="image" hero
|
||||||
|
src={{picture}}></core-image>
|
||||||
|
{{/with}}
|
||||||
{{else}}
|
{{else}}
|
||||||
{{> personaDetailsEdit}}
|
{{#with char}}
|
||||||
|
{{> personaDetailsEdit}}
|
||||||
|
{{/with}}
|
||||||
{{/baseDialog}}
|
{{/baseDialog}}
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template name="personaDetailsEdit">
|
<template name="personaDetailsEdit">
|
||||||
<!--Name-->
|
<div layout horizontal center-justified>
|
||||||
<paper-input id="nameInput" label="Name" floatinglabel value={{name}}></paper-input><br>
|
<div flex style="max-width: 350px;" layout vertical>
|
||||||
<!--Alignment-->
|
<!--Name-->
|
||||||
<paper-input id="alignmentInput" label="Alignment" floatinglabel value={{alignment}}></paper-input><br>
|
<paper-input id="nameInput" label="Name" floatinglabel value={{name}}></paper-input>
|
||||||
<!--Gender-->
|
<!--Alignment-->
|
||||||
<paper-input id="genderInput" label="Gender" floatinglabel value={{gender}}></paper-input><br>
|
<paper-input id="alignmentInput" label="Alignment" floatinglabel value={{alignment}}></paper-input>
|
||||||
<!--Race-->
|
<!--Gender-->
|
||||||
<paper-input id="raceInput" label="Race" floatinglabel value={{race}}></paper-input><br>
|
<paper-input id="genderInput" label="Gender" floatinglabel value={{gender}}></paper-input>
|
||||||
|
<!--Race-->
|
||||||
|
<paper-input id="raceInput" label="Race" floatinglabel value={{race}}></paper-input>
|
||||||
|
<!--Picture-->
|
||||||
|
<paper-input id="pictureInput" label="Picture URL" floatinglabel value={{picture}}></paper-input>
|
||||||
|
<core-image style="height:350px; width: 100%; margin-top: 8px;"
|
||||||
|
sizing="cover"
|
||||||
|
hero-id="image" hero
|
||||||
|
src={{picture}}></core-image>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -2,21 +2,34 @@ Template.personaDetailsEdit.onRendered(function(){
|
|||||||
updatePolymerInputs(this);
|
updatePolymerInputs(this);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
Template.personaDetailsDialog.helpers({
|
||||||
|
char: function() {
|
||||||
|
return Characters.findOne(
|
||||||
|
this._id,
|
||||||
|
{fields: {name: 1, alignment: 1, gender: 1, race: 1, picture: 1}}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
Template.personaDetailsEdit.events({
|
Template.personaDetailsEdit.events({
|
||||||
"change #nameInput": function(event){
|
"change #nameInput": function(event){
|
||||||
var input = event.currentTarget.value;
|
var input = event.currentTarget.value;
|
||||||
Characters.update(this.charId, {$set: {name: input}});
|
Characters.update(this._id, {$set: {name: input}});
|
||||||
},
|
},
|
||||||
"change #alignmentInput": function(event){
|
"change #alignmentInput": function(event){
|
||||||
var input = event.currentTarget.value;
|
var input = event.currentTarget.value;
|
||||||
Characters.update(this.charId, {$set: {alignment: input}});
|
Characters.update(this._id, {$set: {alignment: input}});
|
||||||
},
|
},
|
||||||
"change #genderInput": function(event){
|
"change #genderInput": function(event){
|
||||||
var input = event.currentTarget.value;
|
var input = event.currentTarget.value;
|
||||||
Characters.update(this.charId, {$set: {gender: input}});
|
Characters.update(this._id, {$set: {gender: input}});
|
||||||
},
|
},
|
||||||
"change #raceInput": function(event){
|
"change #raceInput": function(event){
|
||||||
var input = event.currentTarget.value;
|
var input = event.currentTarget.value;
|
||||||
Characters.update(this.charId, {$set: {race: input}});
|
Characters.update(this._id, {$set: {race: input}});
|
||||||
|
},
|
||||||
|
"change #pictureInput": function(event){
|
||||||
|
var input = event.currentTarget.value;
|
||||||
|
Characters.update(this._id, {$set: {picture: input}});
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -3,14 +3,45 @@
|
|||||||
<div id="persona" class="scroll-y" fit>
|
<div id="persona" class="scroll-y" fit>
|
||||||
<div class="column-container">
|
<div class="column-container">
|
||||||
{{#with characterDetails}}
|
{{#with characterDetails}}
|
||||||
{{#containerCardHelper this}}{{alignment}} {{gender}} {{race}}{{/containerCardHelper}}
|
<div>
|
||||||
|
<paper-shadow class="card"
|
||||||
|
hero-id="main" {{detailHero "details" _id}}>
|
||||||
|
{{#unless picture}}
|
||||||
|
<div class="top subhead characterField {{colorClass}}"
|
||||||
|
hero-id="toolbar" {{detailHero "details" _id}}>
|
||||||
|
<div class="subhead" flex
|
||||||
|
hero-id="title" {{detailHero "details" _id}}>
|
||||||
|
{{name}}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{else}}
|
||||||
|
<core-image class="characterField clickable"
|
||||||
|
style="height:350px; width: 100%;
|
||||||
|
background-color: #e8e8e8;"
|
||||||
|
sizing="cover"
|
||||||
|
hero-id="image" {{detailHero "details" _id}}
|
||||||
|
src={{picture}}></core-image>
|
||||||
|
{{/unless}}
|
||||||
|
<div class="bottom">
|
||||||
|
{{#if picture}}
|
||||||
|
<div class="title" hero-id="title" {{detailHero "details" _id}}>
|
||||||
|
{{name}}
|
||||||
|
</div>
|
||||||
|
{{/if}}
|
||||||
|
<div class="subhead">
|
||||||
|
{{alignment}} {{gender}} {{race}}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
{{/with}}
|
{{/with}}
|
||||||
{{> containerCard characterField "description" "Description"}}
|
<div>{{> containerCard characterField "description" "Description"}}</div>
|
||||||
{{> containerCard characterField "personality" "Personality Traits"}}
|
<div>{{> containerCard characterField "personality" "Personality Traits"}}</div>
|
||||||
{{> containerCard characterField "ideals" "Ideals"}}
|
<div>{{> containerCard characterField "ideals" "Ideals"}}</div>
|
||||||
{{> containerCard characterField "bonds" "Bonds"}}
|
<div>{{> containerCard characterField "bonds" "Bonds"}}</div>
|
||||||
{{> containerCard characterField "flaws" "Flaws"}}
|
<div>{{> containerCard characterField "flaws" "Flaws"}}</div>
|
||||||
{{> containerCard characterField "backstory" "Background"}}
|
<div>{{> containerCard characterField "backstory" "Background"}}</div>
|
||||||
|
<div>
|
||||||
<paper-shadow class="card">
|
<paper-shadow class="card">
|
||||||
<div class="white top subhead">
|
<div class="white top subhead">
|
||||||
Languages
|
Languages
|
||||||
@@ -21,6 +52,7 @@
|
|||||||
{{/each}}
|
{{/each}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -40,6 +72,6 @@
|
|||||||
{{title}}
|
{{title}}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="bottom text">{{> UI.contentBlock}}</div>
|
<div class="bottom">{{#markdown}}{{> UI.contentBlock}}{{/markdown}}</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -11,12 +11,12 @@ Template.persona.helpers({
|
|||||||
characterDetails: function(){
|
characterDetails: function(){
|
||||||
var char = Characters.findOne(
|
var char = Characters.findOne(
|
||||||
this._id,
|
this._id,
|
||||||
{fields: {name: 1, gender: 1, alignment: 1, race:1}}
|
{fields: {name: 1, gender: 1, alignment: 1, race:1, picture: 1}}
|
||||||
);
|
);
|
||||||
char.field = "details";
|
char.field = "details";
|
||||||
char.title = char.name;
|
char.title = char.name;
|
||||||
char.color = "d";
|
char.color = "d";
|
||||||
char.topClass = "characterField";
|
char.startEditing = true;
|
||||||
return char;
|
return char;
|
||||||
},
|
},
|
||||||
characterField: function(field, title){
|
characterField: function(field, title){
|
||||||
@@ -40,25 +40,28 @@ Template.persona.helpers({
|
|||||||
|
|
||||||
Template.persona.events({
|
Template.persona.events({
|
||||||
"tap .characterField": function(event){
|
"tap .characterField": function(event){
|
||||||
if (this.field !== "details"){
|
if (this.field == "details"){
|
||||||
var charId = Template.parentData()._id;
|
|
||||||
GlobalUI.setDetail({
|
|
||||||
template: "textDialog",
|
|
||||||
data: {
|
|
||||||
charId: charId,
|
|
||||||
field: this.field,
|
|
||||||
title: this.title,
|
|
||||||
color: this.color,
|
|
||||||
},
|
|
||||||
heroId: this._id + this.field,
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
this.charId = Template.parentData()._id;
|
this.charId = Template.parentData()._id;
|
||||||
GlobalUI.setDetail({
|
GlobalUI.setDetail({
|
||||||
template: "personaDetailsDialog",
|
template: "personaDetailsDialog",
|
||||||
data: this,
|
data: this,
|
||||||
heroId: this._id + this.field,
|
heroId: this._id + this.field,
|
||||||
});
|
});
|
||||||
|
} else {
|
||||||
|
var template = "textDialog";
|
||||||
|
if (this.field === "backstory") template = "backgroundDialog";
|
||||||
|
var charId = Template.parentData()._id;
|
||||||
|
GlobalUI.setDetail({
|
||||||
|
template: template,
|
||||||
|
data: {
|
||||||
|
charId: charId,
|
||||||
|
field: this.field,
|
||||||
|
title: this.title,
|
||||||
|
color: this.color,
|
||||||
|
startEditing: true,
|
||||||
|
},
|
||||||
|
heroId: this._id + this.field,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template name="textDialog">
|
<template name="textDialog">
|
||||||
{{#baseDialog title=title class=colorClass hideColor="true" hideDelete="true" startEditing=startEditing}}
|
{{#baseDialog title=title class=colorClass hideColor="true" hideDelete="true" startEditing=startEditing}}
|
||||||
<div class="pre-wrap">{{evaluateString charId value}}</div>
|
<div>{{#markdown}}{{evaluateString charId value}}{{/markdown}}</div>
|
||||||
{{else}}
|
{{else}}
|
||||||
{{> textDialogEdit}}
|
{{> textDialogEdit}}
|
||||||
{{/baseDialog}}
|
{{/baseDialog}}
|
||||||
|
|||||||
@@ -1,6 +1,10 @@
|
|||||||
<template name="proficiencyView">
|
<template name="proficiencyView">
|
||||||
<div class="proficiencyView" layout horizontal center>
|
<div class="proficiencyView item small"
|
||||||
<core-icon icon="{{profIcon}}"></core-icon>
|
style="padding: 0;"
|
||||||
<div class="sideMargin">{{getName}}</div>
|
layout horizontal center>
|
||||||
|
<core-icon icon="{{profIcon}}" style="margin-right: 16px;"></core-icon>
|
||||||
|
<div>{{getName}}</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
{{#if proficiencies.count}}
|
{{#if proficiencies.count}}
|
||||||
<hr class="vertMargin">
|
<hr class="vertMargin">
|
||||||
<div class="proficiencies">
|
<div class="proficiencies">
|
||||||
<h2 class="spaceAfter">Proficiencies</h2>
|
<h2 style="margin-bottom: 8px;">Proficiencies</h2>
|
||||||
{{#each proficiencies}}
|
{{#each proficiencies}}
|
||||||
{{> proficiencyView}}
|
{{> proficiencyView}}
|
||||||
{{/each}}
|
{{/each}}
|
||||||
|
|||||||
@@ -9,24 +9,33 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template name="spellDetails">
|
<template name="spellDetails">
|
||||||
<div class="caption">
|
<div class="body2">
|
||||||
Level {{level}} {{school}}, {{preparedString}}
|
Level {{level}} {{school}}, {{preparedString}}
|
||||||
</div>
|
</div>
|
||||||
<div class="vertMargin">
|
<div style="margin: 16px 0 16px 0;">
|
||||||
|
{{#if castingTime}}
|
||||||
<div>
|
<div>
|
||||||
<span class="body2">Casting Time: </span><span>{{castingTime}}</span>
|
<span class="body2">Casting Time: </span><span>{{castingTime}}</span>
|
||||||
</div>
|
</div>
|
||||||
|
{{/if}}
|
||||||
|
{{#if range}}
|
||||||
<div>
|
<div>
|
||||||
<span class="body2">Range: </span><span>{{range}}</span>
|
<span class="body2">Range: </span><span>{{range}}</span>
|
||||||
</div>
|
</div>
|
||||||
|
{{/if}}
|
||||||
|
{{#if getComponents}}
|
||||||
<div>
|
<div>
|
||||||
<span class="body2">Components: </span><span>{{getComponents}}</span>
|
<span class="body2">Components: </span><span>{{getComponents}}</span>
|
||||||
</div>
|
</div>
|
||||||
|
{{/if}}
|
||||||
|
{{#if duration}}
|
||||||
<div>
|
<div>
|
||||||
<span class="body2">Duration: </span><span>{{duration}}</span>
|
<span class="body2">Duration: </span><span>{{duration}}</span>
|
||||||
</div>
|
</div>
|
||||||
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
<div class="pre-wrap">{{evaluateString charId description}}</div>
|
<div>{{#markdown}}{{evaluateString charId description}}{{/markdown}}</div>
|
||||||
|
{{> attacksViewList charId=charId parentId=_id}}
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template name="spellEdit">
|
<template name="spellEdit">
|
||||||
@@ -126,4 +135,5 @@
|
|||||||
<textarea id="descriptionInput" placeholder value={{description}}></textarea>
|
<textarea id="descriptionInput" placeholder value={{description}}></textarea>
|
||||||
</paper-autogrow-textarea>
|
</paper-autogrow-textarea>
|
||||||
</paper-input-decorator>
|
</paper-input-decorator>
|
||||||
|
{{> attackEditList parentId=_id parentCollection="Spells" charId=charId enabled=true name=name}}
|
||||||
</template>
|
</template>
|
||||||
@@ -20,7 +20,7 @@
|
|||||||
{{/if}}
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
<hr class="vertMargin">
|
<hr class="vertMargin">
|
||||||
<div class="pre-wrap">{{evaluateString charId description}}</div>
|
<div>{{#markdown}}{{evaluateString charId description}}{{/markdown}}</div>
|
||||||
</div>
|
</div>
|
||||||
{{else}}
|
{{else}}
|
||||||
<!--Name-->
|
<!--Name-->
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<template name="spells">
|
<template name="spells">
|
||||||
<div fit>
|
<div fit>
|
||||||
<div id="spells" class="scroll-y" fit>
|
<div id="spells" class="scroll-y" fit>
|
||||||
<div style="padding: 4px;"
|
<div style="padding: 4px;"
|
||||||
layout horizontal start wrap>
|
layout horizontal start wrap>
|
||||||
{{#if hasSlots}}
|
{{#if hasSlots}}
|
||||||
<paper-shadow class="card"
|
<paper-shadow class="card"
|
||||||
@@ -72,6 +72,7 @@
|
|||||||
<core-tooltip label="Change prepared spells"
|
<core-tooltip label="Change prepared spells"
|
||||||
position="left">
|
position="left">
|
||||||
<paper-icon-button class="prepSpells"
|
<paper-icon-button class="prepSpells"
|
||||||
|
disabled={{#unless canEditCharacter charId}}true{{/unless}}
|
||||||
icon="book">
|
icon="book">
|
||||||
</paper-icon-button>
|
</paper-icon-button>
|
||||||
</core-tooltip>
|
</core-tooltip>
|
||||||
@@ -91,7 +92,7 @@
|
|||||||
<div class="tall spell item"
|
<div class="tall spell item"
|
||||||
hero-id="main" {{detailHero}}
|
hero-id="main" {{detailHero}}
|
||||||
layout horizontal center>
|
layout horizontal center>
|
||||||
<core-icon icon="social:whatshot"
|
<core-icon icon="social:whatshot"
|
||||||
style="color: {{hexColor color}};
|
style="color: {{hexColor color}};
|
||||||
margin-right: 16px;"
|
margin-right: 16px;"
|
||||||
></core-icon>
|
></core-icon>
|
||||||
@@ -143,4 +144,4 @@
|
|||||||
</core-tooltip>
|
</core-tooltip>
|
||||||
{{/fabMenu}}
|
{{/fabMenu}}
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -84,39 +84,35 @@ Template.spells.helpers({
|
|||||||
},
|
},
|
||||||
cantCast: function(level, char){
|
cantCast: function(level, char){
|
||||||
for (var i = level; i <= 9; i++){
|
for (var i = level; i <= 9; i++){
|
||||||
if (char.attributeValue("level" + i + "SpellSlots") > 0){
|
if (Characters.calculate.attributeValue(char._id, "level" + i + "SpellSlots") > 0){
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
baseSlots: function(char){
|
|
||||||
return char.attributeBase("level" + this.level + "SpellSlots");
|
|
||||||
},
|
|
||||||
slots: function(char){
|
|
||||||
return char.attributeValue("level" + this.level + "SpellSlots");
|
|
||||||
},
|
|
||||||
showSlots: function(char){
|
showSlots: function(char){
|
||||||
return this.level && char.attributeBase("level" + this.level + "SpellSlots");
|
return this.level && Characters.calculate.attributeBase(
|
||||||
|
char._id, "level" + this.level + "SpellSlots"
|
||||||
|
);
|
||||||
},
|
},
|
||||||
hasSlots: function(){
|
hasSlots: function(){
|
||||||
for (var i = 1; i <= 9; i += 1){
|
for (var i = 1; i <= 9; i += 1){
|
||||||
if (this.attributeBase("level" + i + "SpellSlots")){
|
if (Characters.calculate.attributeBase(this._id, "level" + i + "SpellSlots")){
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
},
|
},
|
||||||
slotBubbles: function(char){
|
slotBubbles: function(char){
|
||||||
var baseSlots = char.attributeBase("level" + this.level + "SpellSlots");
|
var baseSlots = Characters.calculate.attributeBase(char._id, "level" + this.level + "SpellSlots");
|
||||||
var currentSlots = char.attributeValue("level" + this.level + "SpellSlots");
|
var currentSlots = Characters.calculate.attributeValue(char._id, "level" + this.level + "SpellSlots");
|
||||||
var slotsUsed = baseSlots - currentSlots;
|
var slotsUsed = baseSlots - currentSlots;
|
||||||
var bubbles = [];
|
var bubbles = [];
|
||||||
var i;
|
var i;
|
||||||
for (i = 0; i < currentSlots; i++){
|
for (i = 0; i < currentSlots; i++){
|
||||||
bubbles.push({
|
bubbles.push({
|
||||||
icon: "radio-button-on",
|
icon: "radio-button-on",
|
||||||
disabled: i !== currentSlots - 1, //last full slot not disabled
|
disabled: i !== currentSlots - 1 || !canEditCharacter(char._id), //last full slot not disabled
|
||||||
attribute: "level" + this.level + "SpellSlots",
|
attribute: "level" + this.level + "SpellSlots",
|
||||||
charId: char._id,
|
charId: char._id,
|
||||||
});
|
});
|
||||||
@@ -124,7 +120,7 @@ Template.spells.helpers({
|
|||||||
for (i = 0; i < slotsUsed; i++){
|
for (i = 0; i < slotsUsed; i++){
|
||||||
bubbles.push({
|
bubbles.push({
|
||||||
icon: "radio-button-off",
|
icon: "radio-button-off",
|
||||||
disabled: i !== 0, //first empty slot not disabled
|
disabled: i !== 0 || !canEditCharacter(char._id), //first empty slot not disabled
|
||||||
attribute: "level" + this.level + "SpellSlots",
|
attribute: "level" + this.level + "SpellSlots",
|
||||||
charId: char._id,
|
charId: char._id,
|
||||||
});
|
});
|
||||||
@@ -143,15 +139,15 @@ Template.spells.events({
|
|||||||
var char = Characters.findOne(this.charId);
|
var char = Characters.findOne(this.charId);
|
||||||
if (event.currentTarget.icon === "radio-button-off"){
|
if (event.currentTarget.icon === "radio-button-off"){
|
||||||
if (
|
if (
|
||||||
char.attributeValue(this.attribute) <
|
Characters.calculate.attributeValue(char._id, this.attribute) <
|
||||||
char.attributeBase(this.attribute)
|
Characters.calculate.attributeBase(char._id, this.attribute)
|
||||||
){
|
){
|
||||||
modifier = {$inc: {}};
|
modifier = {$inc: {}};
|
||||||
modifier.$inc[this.attribute + ".adjustment"] = 1;
|
modifier.$inc[this.attribute + ".adjustment"] = 1;
|
||||||
Characters.update(this.charId, modifier, {validate: false});
|
Characters.update(this.charId, modifier, {validate: false});
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (char.attributeValue(this.attribute) > 0){
|
if (Characters.calculate.attributeValue(char._id, this.attribute) > 0){
|
||||||
modifier = {$inc: {}};
|
modifier = {$inc: {}};
|
||||||
modifier.$inc[this.attribute + ".adjustment"] = -1;
|
modifier.$inc[this.attribute + ".adjustment"] = -1;
|
||||||
Characters.update(this.charId, modifier, {validate: false});
|
Characters.update(this.charId, modifier, {validate: false});
|
||||||
|
|||||||
@@ -1,14 +1,16 @@
|
|||||||
<template name="abilityMiniCard">
|
<template name="abilityMiniCard">
|
||||||
<paper-shadow class="card abilityMiniCard clickable"
|
<div>
|
||||||
|
<paper-shadow class="card abilityMiniCard clickable"
|
||||||
hero-id="main" {{detailHero ability ../_id}}
|
hero-id="main" {{detailHero ability ../_id}}
|
||||||
layout horizontal>
|
layout horizontal>
|
||||||
<div class="left white-text {{color}}"
|
<div class="left white-text {{color}}"
|
||||||
hero-id="toolbar" {{detailHero ability ../_id}}>
|
hero-id="toolbar" {{detailHero ability ../_id}}>
|
||||||
<div class="display1">{{../attributeValue ability}}</div>
|
<div class="display1">{{characterCalculate "attributeValue" ../_id ability}}</div>
|
||||||
<div class="title">{{../abilityMod ability}}</div>
|
<div class="title">{{abilityMod}}</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="right subhead" layout horizontal center>
|
<div class="right subhead" layout horizontal center>
|
||||||
{{title}}
|
{{title}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -0,0 +1,9 @@
|
|||||||
|
Template.abilityMiniCard.helpers({
|
||||||
|
abilityMod: function() {
|
||||||
|
return signedString(
|
||||||
|
Characters.calculate.abilityMod(
|
||||||
|
Template.parentData()._id, this.ability
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
@@ -1,9 +1,11 @@
|
|||||||
Template.addTHPDialog.events({
|
Template.addTHPDialog.events({
|
||||||
"tap #addButton": function(event, instance){
|
"tap #addButton": function(event, instance){
|
||||||
|
var max = +instance.find("#quantityInput").value;
|
||||||
|
if (!max || max < 0) max = 0;
|
||||||
TemporaryHitPoints.insert({
|
TemporaryHitPoints.insert({
|
||||||
charId: this.charId,
|
charId: this.charId,
|
||||||
name: instance.find("#nameInput").value,
|
name: instance.find("#nameInput").value,
|
||||||
maximum: +instance.find("#quantityInput").value,
|
maximum: max,
|
||||||
deleteOnZero: !!instance.find("#deleteWhenZeroCheckbox").checked,
|
deleteOnZero: !!instance.find("#deleteWhenZeroCheckbox").checked,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,7 +23,7 @@
|
|||||||
{{#each baseEffects}}
|
{{#each baseEffects}}
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{sourceName}}</td>
|
<td>{{sourceName}}</td>
|
||||||
<td>{{signedString statValue}}</td>
|
<td>Base: {{statValue}}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{{/each}}
|
{{/each}}
|
||||||
{{#each addEffects}}
|
{{#each addEffects}}
|
||||||
@@ -35,7 +35,7 @@
|
|||||||
{{#each mulEffects}}
|
{{#each mulEffects}}
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{sourceName}}</td>
|
<td>{{sourceName}}</td>
|
||||||
<td>×{{statValue}}</td>
|
<td>× {{statValue}}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{{/each}}
|
{{/each}}
|
||||||
{{#each minEffects}}
|
{{#each minEffects}}
|
||||||
|
|||||||
@@ -106,10 +106,8 @@ Template.attributeDialogView.helpers({
|
|||||||
return a || b || c;
|
return a || b || c;
|
||||||
},
|
},
|
||||||
adjustment: function(){
|
adjustment: function(){
|
||||||
var char = Characters.findOne(this.charId);
|
var value = Characters.calculate.attributeValue(this.charId, this.statName);
|
||||||
if (!char) return;
|
var base = Characters.calculate.attributeBase(this.charId, this.statName);
|
||||||
var value = char.attributeValue(this.statName);
|
|
||||||
var base = char.attributeBase(this.statName);
|
|
||||||
return value - base;
|
return value - base;
|
||||||
},
|
},
|
||||||
baseEffects: function(){
|
baseEffects: function(){
|
||||||
@@ -138,14 +136,10 @@ Template.attributeDialogView.helpers({
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
attributeBase: function(){
|
attributeBase: function(){
|
||||||
var char = Characters.findOne(this.charId);
|
return Characters.calculate.attributeBase(this.charId, this.statName);
|
||||||
if (!char) throw "character is " + char;
|
|
||||||
return char.attributeBase(this.statName);
|
|
||||||
},
|
},
|
||||||
attributeValue: function() {
|
attributeValue: function() {
|
||||||
var char = Characters.findOne(this.charId);
|
return Characters.calculate.attributeValue(this.charId, this.statName);
|
||||||
if (!char) throw "character is " + char;
|
|
||||||
return char.attributeValue(this.statName);
|
|
||||||
},
|
},
|
||||||
sourceName: function(){
|
sourceName: function(){
|
||||||
if (this.parent.group === "racial"){
|
if (this.parent.group === "racial"){
|
||||||
|
|||||||
@@ -4,32 +4,7 @@
|
|||||||
<hr class="vertMargin">
|
<hr class="vertMargin">
|
||||||
<div>
|
<div>
|
||||||
<div class="title padded">Carrying</div>
|
<div class="title padded">Carrying</div>
|
||||||
<table class="strengthTable">
|
{{> carryCapacityTable}}
|
||||||
<tr>
|
|
||||||
<td>Encumbered</td>
|
|
||||||
<td>{{evaluate charId "strength * 5"}}lbs</td>
|
|
||||||
<td class="caption">Speed drops by 10 feet</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>Heavily encumbered</td>
|
|
||||||
<td>{{evaluate charId "strength * 10"}}lbs</td>
|
|
||||||
<td class="caption">
|
|
||||||
Speed drops by 20 feet, disadvantage on strength,
|
|
||||||
dexterity and constitution ability checks, attack
|
|
||||||
rolls and saving throws
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>Maximum carrying capacity</td>
|
|
||||||
<td>{{evaluate charId "strength * 15"}}lbs</td>
|
|
||||||
<td class="caption">Speed drops to 5 feet</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>Push, drag or lift maximum</td>
|
|
||||||
<td>{{evaluate charId "strength * 30"}}lbs</td>
|
|
||||||
<td class="caption">You can't move more than this weight</td>
|
|
||||||
</tr>
|
|
||||||
</table>
|
|
||||||
<div class="title padded">Jumping</div>
|
<div class="title padded">Jumping</div>
|
||||||
<table class="strengthTable">
|
<table class="strengthTable">
|
||||||
<tr>
|
<tr>
|
||||||
|
|||||||
@@ -0,0 +1,28 @@
|
|||||||
|
<template name="carryCapacityTable">
|
||||||
|
<table class="carryCapacityTable strengthTable">
|
||||||
|
<tr>
|
||||||
|
<td>Encumbered</td>
|
||||||
|
<td>>{{evaluate charId "strength * 5 * carryMultiplier"}}lbs</td>
|
||||||
|
<td class="caption">Variant rule, encumbered characters move 10 feet slower</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>Heavily encumbered</td>
|
||||||
|
<td>>{{evaluate charId "strength * 10 * carryMultiplier"}}lbs</td>
|
||||||
|
<td class="caption">
|
||||||
|
Variant rule, heavily encumbered characters move 20 feet slower and have disadvantage on ability checks, attack rolls, and saving thows that use Strength, Dexterity, or Constitution
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>Over Encumbered</td>
|
||||||
|
<td>>{{evaluate charId "strength * 15 * carryMultiplier"}}lbs</td>
|
||||||
|
<td class="caption">
|
||||||
|
Characters that can only just lift, push or drag their current load can only move at 5 feet.
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>Push, drag or lift maximum</td>
|
||||||
|
<td>{{evaluate charId "strength * 30 * carryMultiplier"}}lbs</td>
|
||||||
|
<td class="caption"></td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
</template>
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
.healthCard paper-slider{
|
.healthCard paper-diff-slider{
|
||||||
width: 100%;
|
width: 100%;
|
||||||
margin-right: 8px;
|
margin-right: 8px;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,43 +1,60 @@
|
|||||||
<template name="healthCard">
|
<template name="healthCard">
|
||||||
<paper-shadow class="card container healthCard"
|
<paper-shadow class="card container healthCard"
|
||||||
hero-id="main" {{detailHero "hitPoints" _id}}
|
hero-id="main" {{detailHero "hitPoints" _id}}
|
||||||
layout horizontal wrap>
|
layout horizontal wrap>
|
||||||
<div class="green white-text subhead left"
|
<div class="green white-text subhead left"
|
||||||
hero-id="toolbar" {{detailHero "hitPoints" _id}}
|
hero-id="toolbar" {{detailHero "hitPoints" _id}}
|
||||||
layout vertical center center-justified>
|
layout vertical center center-justified>
|
||||||
<div class="hitPointTitle clickable">Hit Points</div>
|
<div class="hitPointTitle clickable">Hit Points</div>
|
||||||
<paper-icon-button class="white54" id="addTempHP" icon="add"></paper-icon-button>
|
<paper-icon-button class="white54"
|
||||||
|
id="addTempHP"
|
||||||
|
icon="add"
|
||||||
|
disabled={{#unless canEditCharacter _id}}true{{/unless}}>
|
||||||
|
</paper-icon-button>
|
||||||
</div>
|
</div>
|
||||||
<div class="right" flex layout vertical center-justified style="min-width: 180px;">
|
<div class="right" flex layout vertical center-justified style="min-width: 180px;">
|
||||||
<div layout horizontal>
|
<div layout horizontal>
|
||||||
<paper-slider id="hitPointSlider"
|
<paper-diff-slider id="hitPointSlider"
|
||||||
value={{attributeValue "hitPoints"}}
|
value={{characterCalculate "attributeValue" _id "hitPoints"}}
|
||||||
max={{attributeBase "hitPoints"}}
|
max={{characterCalculate "attributeBase" _id "hitPoints"}}
|
||||||
editable pin
|
editable pin
|
||||||
role="slider"
|
disabled={{#unless canEditCharacter _id}}true{{/unless}}
|
||||||
></paper-slider>
|
role="slider">
|
||||||
|
</paper-diff-slider>
|
||||||
</div>
|
</div>
|
||||||
{{#each tempHitPoints}}
|
{{#each tempHitPoints}}
|
||||||
<div>
|
<div>
|
||||||
{{name}}
|
{{name}}
|
||||||
<div layout horizontal>
|
<div layout horizontal>
|
||||||
<paper-slider class="tempHitPointSlider"
|
<paper-diff-slider class="tempHitPointSlider"
|
||||||
value={{left}}
|
value={{left}}
|
||||||
max={{maximum}}
|
max={{maximum}}
|
||||||
editable pin
|
editable pin
|
||||||
role="slider"
|
role="slider"
|
||||||
flex
|
flex
|
||||||
></paper-slider>
|
></paper-diff-slider>
|
||||||
{{#unless left}}{{#unless deleteOnZero}}
|
{{#unless left}}
|
||||||
<paper-icon-button class="deleteTHP" icon="delete"></paper-icon-button>
|
<paper-icon-button class="deleteTHP" icon="delete"></paper-icon-button>
|
||||||
{{/unless}}{{/unless}}
|
{{/unless}}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{{/each}}
|
{{/each}}
|
||||||
<div class="caption">
|
<div class="caption">
|
||||||
{{#if multipliers.immunities.length}} <div>Immune: {{#each multipliers.immunities}} {{name}} {{/each}}</div>{{/if}}
|
{{#if multipliers.immunities.length}}
|
||||||
{{#if multipliers.resistances.length}}<div>Resistance: {{#each multipliers.resistances}} {{name}} {{/each}}</div>{{/if}}
|
<div>
|
||||||
{{#if multipliers.weaknesses.length}} <div>Weakness: {{#each multipliers.weaknesses}} {{name}} {{/each}}</div>{{/if}}
|
Immune: {{#each multipliers.immunities}} {{name}} {{/each}}
|
||||||
|
</div>
|
||||||
|
{{/if}}
|
||||||
|
{{#if multipliers.resistances.length}}
|
||||||
|
<div>
|
||||||
|
Resistance: {{#each multipliers.resistances}} {{name}} {{/each}}
|
||||||
|
</div>
|
||||||
|
{{/if}}
|
||||||
|
{{#if multipliers.weaknesses.length}}
|
||||||
|
<div>
|
||||||
|
Weakness: {{#each multipliers.weaknesses}} {{name}} {{/each}}
|
||||||
|
</div>
|
||||||
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
{{#if showDeathSave}}
|
{{#if showDeathSave}}
|
||||||
{{#with deathSaveObject}}
|
{{#with deathSaveObject}}
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ Template.healthCard.helpers({
|
|||||||
return TemporaryHitPoints.find({charId: this._id});
|
return TemporaryHitPoints.find({charId: this._id});
|
||||||
},
|
},
|
||||||
showDeathSave: function(){
|
showDeathSave: function(){
|
||||||
return this.attributeValue("hitPoints") <= 0;
|
return Characters.calculate.attributeValue(this._id, "hitPoints") <= 0;
|
||||||
},
|
},
|
||||||
deathSaveObject: function(){
|
deathSaveObject: function(){
|
||||||
var char = Characters.findOne(this._id, {fields: {deathSave: 1}});
|
var char = Characters.findOne(this._id, {fields: {deathSave: 1}});
|
||||||
@@ -27,21 +27,20 @@ Template.healthCard.helpers({
|
|||||||
return this.fail >= 3;
|
return this.fail >= 3;
|
||||||
},
|
},
|
||||||
multipliers: function(){
|
multipliers: function(){
|
||||||
var char = Characters.findOne(this._id, {fields: {_id: 1}});
|
|
||||||
var multipliers = [
|
var multipliers = [
|
||||||
{name: "Acid", value: char.attributeValue("acidMultiplier", 1)},
|
{name: "Acid", value: Characters.calculate.attributeValue(this._id, "acidMultiplier")},
|
||||||
{name: "Bludgeoning", value: char.attributeValue("bludgeoningMultiplier", 1)},
|
{name: "Bludgeoning", value: Characters.calculate.attributeValue(this._id, "bludgeoningMultiplier")},
|
||||||
{name: "Cold", value: char.attributeValue("coldMultiplier", 1)},
|
{name: "Cold", value: Characters.calculate.attributeValue(this._id, "coldMultiplier")},
|
||||||
{name: "Fire", value: char.attributeValue("fireMultiplier", 1)},
|
{name: "Fire", value: Characters.calculate.attributeValue(this._id, "fireMultiplier")},
|
||||||
{name: "Force", value: char.attributeValue("forceMultiplier", 1)},
|
{name: "Force", value: Characters.calculate.attributeValue(this._id, "forceMultiplier")},
|
||||||
{name: "Lightning", value: char.attributeValue("lightningMultiplier", 1)},
|
{name: "Lightning", value: Characters.calculate.attributeValue(this._id, "lightningMultiplier")},
|
||||||
{name: "Necrotic", value: char.attributeValue("necroticMultiplier", 1)},
|
{name: "Necrotic", value: Characters.calculate.attributeValue(this._id, "necroticMultiplier")},
|
||||||
{name: "Piercing", value: char.attributeValue("piercingMultiplier", 1)},
|
{name: "Piercing", value: Characters.calculate.attributeValue(this._id, "piercingMultiplier")},
|
||||||
{name: "Poison", value: char.attributeValue("poisonMultiplier", 1)},
|
{name: "Poison", value: Characters.calculate.attributeValue(this._id, "poisonMultiplier")},
|
||||||
{name: "Psychic", value: char.attributeValue("psychicMultiplier", 1)},
|
{name: "Psychic", value: Characters.calculate.attributeValue(this._id, "psychicMultiplier")},
|
||||||
{name: "Radiant", value: char.attributeValue("radiantMultiplier", 1)},
|
{name: "Radiant", value: Characters.calculate.attributeValue(this._id, "radiantMultiplier")},
|
||||||
{name: "Slashing", value: char.attributeValue("slashingMultiplier", 1)},
|
{name: "Slashing", value: Characters.calculate.attributeValue(this._id, "slashingMultiplier")},
|
||||||
{name: "Thunder", value: char.attributeValue("thunderMultiplier", 1)},
|
{name: "Thunder", value: Characters.calculate.attributeValue(this._id, "thunderMultiplier")},
|
||||||
];
|
];
|
||||||
multipliers = _.groupBy(multipliers, "value");
|
multipliers = _.groupBy(multipliers, "value");
|
||||||
return {
|
return {
|
||||||
@@ -55,7 +54,8 @@ Template.healthCard.helpers({
|
|||||||
Template.healthCard.events({
|
Template.healthCard.events({
|
||||||
"change #hitPointSlider": function(event){
|
"change #hitPointSlider": function(event){
|
||||||
var value = event.currentTarget.value;
|
var value = event.currentTarget.value;
|
||||||
var adjustment = value - this.attributeBase("hitPoints");
|
var base = Characters.calculate.attributeBase(this._id, "hitPoints")
|
||||||
|
var adjustment = value - base;
|
||||||
Characters.update(this._id, {$set: {"hitPoints.adjustment": adjustment}});
|
Characters.update(this._id, {$set: {"hitPoints.adjustment": adjustment}});
|
||||||
//reset the death saves if we are gaining HP
|
//reset the death saves if we are gaining HP
|
||||||
if (value > 0)
|
if (value > 0)
|
||||||
|
|||||||
@@ -1,14 +1,15 @@
|
|||||||
<template name="hitDice">
|
<template name="hitDice">
|
||||||
{{#if ../attributeBase name}}
|
{{#if characterCalculate "attributeBase" ../_id name}}
|
||||||
<paper-shadow class="card hit-dice" hero-id="main"
|
<div>
|
||||||
{{detailHero name ../_id}}
|
<paper-shadow class="card hit-dice" hero-id="main"
|
||||||
|
{{detailHero name ../_id}}
|
||||||
layout horizontal>
|
layout horizontal>
|
||||||
<div class="left green display1 white-text"
|
<div class="left green display1 white-text"
|
||||||
hero-id="toolbar" {{detailHero name ../_id}}
|
hero-id="toolbar" {{detailHero name ../_id}}
|
||||||
layout horizontal>
|
layout horizontal>
|
||||||
<div>
|
<div>
|
||||||
<paper-icon-button class="resourceUp"
|
<paper-icon-button class="resourceUp"
|
||||||
icon="arrow-drop-up"
|
icon="arrow-drop-up"
|
||||||
disabled={{cantIncrement}}>
|
disabled={{cantIncrement}}>
|
||||||
</paper-icon-button>
|
</paper-icon-button>
|
||||||
<paper-icon-button class="resourceDown"
|
<paper-icon-button class="resourceDown"
|
||||||
@@ -18,10 +19,10 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="resourceValue" layout vertical center>
|
<div class="resourceValue" layout vertical center>
|
||||||
<div>
|
<div>
|
||||||
{{../attributeValue name}}
|
{{characterCalculate "attributeValue" ../_id name}}
|
||||||
</div>
|
</div>
|
||||||
<div class="title white-text">
|
<div class="title white-text">
|
||||||
d{{diceNum}} {{../abilityMod "constitution"}}
|
d{{diceNum}} {{conMod}}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -29,5 +30,6 @@
|
|||||||
Hit Dice
|
Hit Dice
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -1,25 +1,33 @@
|
|||||||
Template.hitDice.helpers({
|
Template.hitDice.helpers({
|
||||||
cantIncrement: function(){
|
cantIncrement: function(){
|
||||||
var valueSmallerThanBase = this.char.attributeValue(this.name) <
|
var value = Characters.calculate.attributeValue(this.char._id, this.name);
|
||||||
this.char.attributeBase(this.name);
|
var base = Characters.calculate.attributeBase(this.char._id, this.name);
|
||||||
return !valueSmallerThanBase;
|
return value >= base || !canEditCharacter(this.char._id);
|
||||||
},
|
},
|
||||||
cantDecrement: function(){
|
cantDecrement: function(){
|
||||||
var valuePositive = this.char.attributeValue(this.name) > 0;
|
var value = Characters.calculate.attributeValue(this.char._id, this.name);
|
||||||
return !valuePositive;
|
return value <= 0 || !canEditCharacter(this.char._id);
|
||||||
|
},
|
||||||
|
conMod: function(){
|
||||||
|
return signedString(
|
||||||
|
Characters.calculate.abilityMod(this.char._id, "constitution")
|
||||||
|
);
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
Template.hitDice.events({
|
Template.hitDice.events({
|
||||||
"tap .resourceUp": function(event){
|
"tap .resourceUp": function(event){
|
||||||
if (this.char.attributeValue(this.name) < this.char.attributeBase(this.name)){
|
var value = Characters.calculate.attributeValue(this.char._id, this.name);
|
||||||
|
var base = Characters.calculate.attributeBase(this.char._id, this.name);
|
||||||
|
if (value < base){
|
||||||
var modifier = {$inc: {}};
|
var modifier = {$inc: {}};
|
||||||
modifier.$inc[this.name + ".adjustment"] = 1;
|
modifier.$inc[this.name + ".adjustment"] = 1;
|
||||||
Characters.update(this.char._id, modifier, {validate: false});
|
Characters.update(this.char._id, modifier, {validate: false});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"tap .resourceDown": function(event){
|
"tap .resourceDown": function(event){
|
||||||
if (this.char.attributeValue(this.name) > 0){
|
var value = Characters.calculate.attributeValue(this.char._id, this.name);
|
||||||
|
if (value > 0){
|
||||||
var modifier = {$inc: {}};
|
var modifier = {$inc: {}};
|
||||||
modifier.$inc[this.name + ".adjustment"] = -1;
|
modifier.$inc[this.name + ".adjustment"] = -1;
|
||||||
Characters.update(this.char._id, modifier, {validate: false});
|
Characters.update(this.char._id, modifier, {validate: false});
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
<!-- needs name, char, and skillName -->
|
<!-- needs name, charId, and skillName -->
|
||||||
<template name="skillDialog">
|
<template name="skillDialog">
|
||||||
{{#baseDialog title=name class=color hideEdit=true}}
|
{{#baseDialog title=name class=color hideEdit=true}}
|
||||||
{{> skillDialogView}}
|
{{> skillDialogView}}
|
||||||
@@ -8,7 +8,7 @@
|
|||||||
<template name="skillDialogView">
|
<template name="skillDialogView">
|
||||||
<div layout vertical center>
|
<div layout vertical center>
|
||||||
<div class="display2">
|
<div class="display2">
|
||||||
{{char.skillMod skillName}}
|
{{characterCalculate "skillMod" charId skillName}}
|
||||||
</div>
|
</div>
|
||||||
<div class="subhead">
|
<div class="subhead">
|
||||||
<core-icon icon="{{profIcon}}" class="black54"></core-icon>
|
<core-icon icon="{{profIcon}}" class="black54"></core-icon>
|
||||||
@@ -25,9 +25,9 @@
|
|||||||
<table class="summaryTable">
|
<table class="summaryTable">
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{abilityName}}</td>
|
<td>{{abilityName}}</td>
|
||||||
<td>{{char.abilityMod ability}}</td>
|
<td>{{characterCalculate "abilityMod" charId ability}}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{{#if char.proficiency skillName}}
|
{{#if characterCalculate "proficiency" charId skillName}}
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{proficiencyValue}}</td>
|
<td>{{proficiencyValue}}</td>
|
||||||
<td>{{signedString profBonus}}</td>
|
<td>{{signedString profBonus}}</td>
|
||||||
@@ -59,7 +59,7 @@
|
|||||||
{{/each}}
|
{{/each}}
|
||||||
<tr class="body2">
|
<tr class="body2">
|
||||||
<td>Total</td>
|
<td>Total</td>
|
||||||
<td>{{char.skillMod skillName}}</td>
|
<td>{{characterCalculate "skillMod" charId skillName}}</td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
|
|||||||
@@ -106,9 +106,7 @@ Template.skillDialogView.helpers({
|
|||||||
return a || b || c;
|
return a || b || c;
|
||||||
},
|
},
|
||||||
profIcon: function(){
|
profIcon: function(){
|
||||||
var char = Characters.findOne(this.charId);
|
var prof = Characters.calculate.proficiency(this.charId, this.skillName);
|
||||||
if (!char) return;
|
|
||||||
var prof = char.proficiency(this.skillName);
|
|
||||||
if (prof > 0 && prof < 1) return "image:brightness-2";
|
if (prof > 0 && prof < 1) return "image:brightness-2";
|
||||||
if (prof === 1) return "image:brightness-1";
|
if (prof === 1) return "image:brightness-1";
|
||||||
if (prof > 1) return "av:album";
|
if (prof > 1) return "av:album";
|
||||||
@@ -123,13 +121,13 @@ Template.skillDialogView.helpers({
|
|||||||
profBonus: function(){
|
profBonus: function(){
|
||||||
var char = Characters.findOne(this.charId);
|
var char = Characters.findOne(this.charId);
|
||||||
if (!char) return;
|
if (!char) return;
|
||||||
return char.proficiency(this.skillName) *
|
var prof = Characters.calculate.proficiency(this.charId, this.skillName);
|
||||||
char.attributeValue("proficiencyBonus");
|
var proficiencyBonus =
|
||||||
|
Characters.calculate.attributeValue(this.charId, "proficiencyBonus");
|
||||||
|
return prof * proficiencyBonus;
|
||||||
},
|
},
|
||||||
proficiencyValue: function(){
|
proficiencyValue: function(){
|
||||||
var char = Characters.findOne(this.charId);
|
var prof = Characters.calculate.proficiency(this.charId, this.skillName);
|
||||||
if (!char) return;
|
|
||||||
var prof = char.proficiency(this.skillName);
|
|
||||||
if (prof == 0.5) return "Half Proficiency";
|
if (prof == 0.5) return "Half Proficiency";
|
||||||
if (prof == 1) return "Proficient";
|
if (prof == 1) return "Proficient";
|
||||||
if (prof == 2) return "Double Proficiency";
|
if (prof == 2) return "Double Proficiency";
|
||||||
@@ -199,20 +197,21 @@ Template.skillDialogView.helpers({
|
|||||||
return skill.ability;
|
return skill.ability;
|
||||||
},
|
},
|
||||||
abilityName: function(){
|
abilityName: function(){
|
||||||
var opts = {fields: {}};
|
var skill = Characters.calculate.getField(this.charId, this.skillName);
|
||||||
opts.fields[this.skillName] = 1;
|
|
||||||
var char = Characters.findOne(this.charId, opts);
|
|
||||||
if (!char) return;
|
|
||||||
var skill = char[this.skillName];
|
|
||||||
if (!skill) return;
|
if (!skill) return;
|
||||||
var ability = skill.ability;
|
var ability = skill.ability;
|
||||||
return abilities[ability] && abilities[ability].name;
|
return abilities[ability] && abilities[ability].name;
|
||||||
},
|
},
|
||||||
char: function(){
|
|
||||||
return Characters.findOne(this.charId, {fields:{_id: 1}});
|
|
||||||
},
|
|
||||||
sourceName: function(){
|
sourceName: function(){
|
||||||
if (this.parent.collection === "Characters") return "inate";
|
if (this.parent.collection === "Characters"){
|
||||||
|
if (this.parent.group === "racial"){
|
||||||
|
return Characters.calculate.getField(this.charId, "race") || "Race";
|
||||||
|
}
|
||||||
|
if (this.parent.group === "background"){
|
||||||
|
return "Background";
|
||||||
|
}
|
||||||
|
return "Innate";
|
||||||
|
}
|
||||||
return this.getParent().name;
|
return this.getParent().name;
|
||||||
},
|
},
|
||||||
operationName: function(){
|
operationName: function(){
|
||||||
|
|||||||
@@ -8,7 +8,9 @@
|
|||||||
{{#if failSkill}}
|
{{#if failSkill}}
|
||||||
<div class="fail skill-mod">fail</div>
|
<div class="fail skill-mod">fail</div>
|
||||||
{{else}}
|
{{else}}
|
||||||
<div class="{{advantage}} skill-mod">{{../skillMod skill}}</div>
|
<div class="{{advantage}} skill-mod">
|
||||||
|
{{skillMod}}
|
||||||
|
</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
<div flex>
|
<div flex>
|
||||||
{{name}}
|
{{name}}
|
||||||
@@ -16,7 +18,7 @@
|
|||||||
*
|
*
|
||||||
{{/if}}
|
{{/if}}
|
||||||
{{#if showPassive}}
|
{{#if showPassive}}
|
||||||
({{../passiveSkill skill}})
|
({{characterCalculate "passiveSkill" ../_id skill}})
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,13 +1,21 @@
|
|||||||
Template.skillRow.helpers({
|
Template.skillRow.helpers({
|
||||||
|
skillMod: function() {
|
||||||
|
return signedString(
|
||||||
|
Characters.calculate.skillMod(
|
||||||
|
Template.parentData()._id, this.skill
|
||||||
|
)
|
||||||
|
);
|
||||||
|
},
|
||||||
profIcon: function(){
|
profIcon: function(){
|
||||||
var prof = Template.parentData(1).proficiency(this.skill);
|
var charId = Template.parentData()._id;
|
||||||
|
var prof = Characters.calculate.proficiency(charId, this.skill);
|
||||||
if (prof > 0 && prof < 1) return "image:brightness-2";
|
if (prof > 0 && prof < 1) return "image:brightness-2";
|
||||||
if (prof === 1) return "image:brightness-1";
|
if (prof === 1) return "image:brightness-1";
|
||||||
if (prof > 1) return "av:album";
|
if (prof > 1) return "av:album";
|
||||||
return "radio-button-off";
|
return "radio-button-off";
|
||||||
},
|
},
|
||||||
failSkill: function(){
|
failSkill: function(){
|
||||||
var charId = Template.parentData(1)._id;
|
var charId = Template.parentData()._id;
|
||||||
return Effects.find({
|
return Effects.find({
|
||||||
charId: charId,
|
charId: charId,
|
||||||
stat: this.skill,
|
stat: this.skill,
|
||||||
@@ -16,12 +24,13 @@ Template.skillRow.helpers({
|
|||||||
}).count();
|
}).count();
|
||||||
},
|
},
|
||||||
advantage: function(){
|
advantage: function(){
|
||||||
var advantage = Template.parentData(1).advantage(this.skill);
|
var charId = Template.parentData()._id;
|
||||||
|
var advantage = Characters.calculate.advantage(charId, this.skill);
|
||||||
if (advantage > 0) return "advantage";
|
if (advantage > 0) return "advantage";
|
||||||
if (advantage < 0) return "disadvantage";
|
if (advantage < 0) return "disadvantage";
|
||||||
},
|
},
|
||||||
conditionalCount: function(){
|
conditionalCount: function(){
|
||||||
var charId = Template.parentData(1)._id;
|
var charId = Template.parentData()._id;
|
||||||
return Effects.find({
|
return Effects.find({
|
||||||
charId: charId,
|
charId: charId,
|
||||||
stat: this.skill,
|
stat: this.skill,
|
||||||
|
|||||||
@@ -27,6 +27,7 @@
|
|||||||
{{>hitDice name="d10HitDice" diceNum="10" char=this}}
|
{{>hitDice name="d10HitDice" diceNum="10" char=this}}
|
||||||
{{>hitDice name="d12HitDice" diceNum="12" char=this}}
|
{{>hitDice name="d12HitDice" diceNum="12" char=this}}
|
||||||
<!--Saving Throws-->
|
<!--Saving Throws-->
|
||||||
|
<div>
|
||||||
<paper-shadow class="card">
|
<paper-shadow class="card">
|
||||||
<div class="top white subhead">
|
<div class="top white subhead">
|
||||||
Saving Throws
|
Saving Throws
|
||||||
@@ -40,7 +41,9 @@
|
|||||||
{{> skillRow name="Charisma" skill="charismaSave"}}
|
{{> skillRow name="Charisma" skill="charismaSave"}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
<!--Skills-->
|
<!--Skills-->
|
||||||
|
<div>
|
||||||
<paper-shadow class="card">
|
<paper-shadow class="card">
|
||||||
<div class="top white subhead">
|
<div class="top white subhead">
|
||||||
Skills
|
Skills
|
||||||
@@ -66,22 +69,25 @@
|
|||||||
{{> skillRow name="Survival" skill="survival"}}
|
{{> skillRow name="Survival" skill="survival"}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template name="statCard">
|
<template name="statCard">
|
||||||
|
<div>
|
||||||
<paper-shadow class="card statCard clickable" hero-id="main" {{detailHero stat ../_id}} layout horizontal>
|
<paper-shadow class="card statCard clickable" hero-id="main" {{detailHero stat ../_id}} layout horizontal>
|
||||||
<div class="left display1 white-text {{color}}"
|
<div class="left display1 white-text {{color}}"
|
||||||
hero-id="toolbar" {{detailHero stat ../_id}}>
|
hero-id="toolbar" {{detailHero stat ../_id}}>
|
||||||
{{#if isSkill}}
|
{{#if isSkill}}
|
||||||
{{../skillMod stat}}
|
{{prefix}}{{skillMod}}
|
||||||
{{else}}
|
{{else}}
|
||||||
{{prefix}}{{../attributeValue stat}}
|
{{prefix}}{{characterCalculate "attributeValue" ../_id stat}}
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
<div class="right subhead" flex horizontal layout center>
|
<div class="right subhead" flex horizontal layout center>
|
||||||
{{name}}
|
{{name}}
|
||||||
</div>
|
</div>
|
||||||
</paper-shadow>
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -67,6 +67,12 @@ Template.stats.events({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
Template.stats.helpers({
|
Template.statCard.helpers({
|
||||||
|
skillMod: function() {
|
||||||
|
return signedString(
|
||||||
|
Characters.calculate.skillMod(
|
||||||
|
Template.parentData()._id, this.stat
|
||||||
|
)
|
||||||
|
);
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<template name="characterList">
|
<template name="characterList">
|
||||||
<core-toolbar class="blue-grey white-text">
|
<core-toolbar class="app-grey white-text">
|
||||||
<core-icon-button icon="menu" core-drawer-toggle></core-icon-button>
|
<core-icon-button icon="menu" core-drawer-toggle></core-icon-button>
|
||||||
<div flex>
|
<div flex>
|
||||||
Characters
|
Characters
|
||||||
|
|||||||
@@ -13,7 +13,10 @@ Template.characterSideList.helpers({
|
|||||||
{owner: userId},
|
{owner: userId},
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{fields: {name: 1}}
|
{
|
||||||
|
fields: {name: 1},
|
||||||
|
sort: {name: 1},
|
||||||
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -25,6 +25,13 @@
|
|||||||
</paper-autogrow-textarea>
|
</paper-autogrow-textarea>
|
||||||
</paper-input-decorator>
|
</paper-input-decorator>
|
||||||
</div>
|
</div>
|
||||||
<paper-button id="cancelButton" affirmative>Cancel</paper-button>
|
<paper-button id="cancelButton"
|
||||||
<paper-button id="sendButton" affirmative>Send </paper-button>
|
affirmative>
|
||||||
|
Cancel
|
||||||
|
</paper-button>
|
||||||
|
<paper-button id="sendButton"
|
||||||
|
affirmative
|
||||||
|
disabled={{invalid}}>
|
||||||
|
Send
|
||||||
|
</paper-button>
|
||||||
</template>
|
</template>
|
||||||
@@ -1,4 +1,23 @@
|
|||||||
|
Template.feedback.onCreated(function() {
|
||||||
|
this.title = new ReactiveVar("");
|
||||||
|
this.description = new ReactiveVar("");
|
||||||
|
});
|
||||||
|
|
||||||
|
Template.feedback.helpers({
|
||||||
|
invalid: function() {
|
||||||
|
var inst = Template.instance();
|
||||||
|
return !inst.title.get() ||
|
||||||
|
!inst.description.get();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
Template.feedback.events({
|
Template.feedback.events({
|
||||||
|
"input #feedbackTitle": function(event, instance) {
|
||||||
|
instance.title.set(instance.find("#feedbackTitle").value);
|
||||||
|
},
|
||||||
|
"input #feedbackDescription": function(event, instance) {
|
||||||
|
instance.description.set(instance.find("#feedbackDescription").value);
|
||||||
|
},
|
||||||
"tap #sendButton": function(event, instance) {
|
"tap #sendButton": function(event, instance) {
|
||||||
var report = {};
|
var report = {};
|
||||||
report.title = instance.find("#feedbackTitle").value;
|
report.title = instance.find("#feedbackTitle").value;
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
<template name="guide">
|
<template name="guide">
|
||||||
|
<core-toolbar class="app-grey white-text">
|
||||||
|
<core-icon-button icon="menu" core-drawer-toggle></core-icon-button>
|
||||||
|
<div flex>Guide</div>
|
||||||
|
</core-toolbar>
|
||||||
<div layout vertical center>
|
<div layout vertical center>
|
||||||
<paper-shadow class="wallOfText card" style="padding: 32px; max-width: 800px;">
|
<paper-shadow class="wallOfText card" style="padding: 32px; max-width: 800px;">
|
||||||
<h1>Dicecloud Beta</h1>
|
|
||||||
<p>Welcome to the Dicecloud beta. Please don't share the link with people you don't actively play with, since the beta is intended to be small, and your experience will probably get laggy if it gets more traffic than I'm expecting.</p>
|
|
||||||
<p>The beta is going to start with just the character sheet. You can play D&D without minis and maps, without a pre-written adventure, you can play without a lot of things, but the character sheet is necessary. So I'm starting here and working my way outwards.</p>
|
|
||||||
<p>I will eventually have public bug tracking and feature requests going, but for now I'm going to track comments, feedback and suggestions on <a href="http://reddit.com/r/dicecloud">this subreddit</a>. If you've never used reddit before, all you need is a username and password to sign up. So it should be pretty accessible.</p>
|
|
||||||
<h2>Character Sheet Philosophy</h2>
|
<h2>Character Sheet Philosophy</h2>
|
||||||
<p>Setting up your character on Dicecloud is going to take you a little longer than just filling it in on a paper character sheet would have. The goal of using an online sheet is to make actually playing the game more streamlined, and ultimately more fun. So putting a little extra effort into setting up your character now will pay off over and over again once you're playing.</p>
|
<p>Setting up your character on Dicecloud is going to take you a little longer than just filling it in on a paper character sheet would have. The goal of using an online sheet is to make actually playing the game more streamlined, and ultimately more fun. So putting a little extra effort into setting up your character now will pay off over and over again once you're playing.</p>
|
||||||
<p>The idea is to track where each number comes from, and allow you to easily make changes on the fly.</p>
|
<p>The idea is to track where each number comes from, and allow you to easily make changes on the fly.</p>
|
||||||
|
|||||||
@@ -1,11 +1,9 @@
|
|||||||
<template name="home">
|
<template name="home">
|
||||||
<core-toolbar class="blue-grey white-text">
|
<core-toolbar class="app-grey white-text">
|
||||||
<core-icon-button icon="menu" core-drawer-toggle></core-icon-button>
|
<core-icon-button icon="menu" core-drawer-toggle></core-icon-button>
|
||||||
<div flex>
|
<div flex>
|
||||||
DiceCloud
|
Home
|
||||||
</div>
|
</div>
|
||||||
</core-toolbar>
|
</core-toolbar>
|
||||||
<div class="scroll-y" style="padding: 16px" fit>
|
{{> intro}}
|
||||||
{{> guide}}
|
|
||||||
</div>
|
|
||||||
</template>
|
</template>
|
||||||
@@ -1,25 +1,15 @@
|
|||||||
Template.home.helpers({
|
Template.home.helpers({
|
||||||
characterDetails: function(){
|
selectedTab: function(){
|
||||||
var char = Characters.findOne(
|
return Session.get("homePage.selectedTab");
|
||||||
this._id,
|
},
|
||||||
{fields: {name: 1, gender: 1, alignment: 1, race:1}}
|
|
||||||
);
|
|
||||||
char.title = char.name;
|
|
||||||
char.field = "base";
|
|
||||||
char.color = "d";
|
|
||||||
char.class = "characterCard";
|
|
||||||
return char;
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
Template.home.events({
|
Template.home.events({
|
||||||
"tap .characterCard": function(event, instance){
|
"core-animated-pages-transition-end .tabPages": function(event) {
|
||||||
Router.go("characterSheet", {_id: this._id});
|
event.stopPropagation();
|
||||||
},
|
},
|
||||||
"tap #addCharacter": function(event, template) {
|
"tap .homeTabs paper-tab": function(event, instance){
|
||||||
Characters.insert({owner: Meteor.userId()});
|
Session.set("homePage.selectedTab",
|
||||||
},
|
event.currentTarget.getAttribute("name"));
|
||||||
"tap #deleteChar": function(event, template){
|
|
||||||
Characters.remove(this._id);
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
98
rpg-docs/client/views/home/intro/intro.html
Normal file
98
rpg-docs/client/views/home/intro/intro.html
Normal file
@@ -0,0 +1,98 @@
|
|||||||
|
<template name="intro">
|
||||||
|
<div class="intro">
|
||||||
|
<div class="section white-text" style="background: #282828">
|
||||||
|
<div class="display2">
|
||||||
|
Dice Cloud
|
||||||
|
</div>
|
||||||
|
<img style="width:130px; height:130px; background-color: #282828;"
|
||||||
|
src="/crown-dice-logo-cropped-transparent.png">
|
||||||
|
<div class="display1">
|
||||||
|
Unofficial Online Realtime D&D 5e App
|
||||||
|
</div>
|
||||||
|
<h2>
|
||||||
|
Spend less time shuffling paper and more time playing the game
|
||||||
|
</h2>
|
||||||
|
{{#unless currentUser}}
|
||||||
|
<div layout horizontal around-justified wrap>
|
||||||
|
<paper-button class="red white-text signInButton"
|
||||||
|
style="margin: 16px;"
|
||||||
|
raised>
|
||||||
|
Sign In
|
||||||
|
</paper-button>
|
||||||
|
<paper-button class="red white-text signUpButton"
|
||||||
|
style="margin: 16px;"
|
||||||
|
raised>
|
||||||
|
Sign Up
|
||||||
|
</paper-button>
|
||||||
|
</div>
|
||||||
|
{{else}}
|
||||||
|
<div style="padding-bottom: 0;"></div>
|
||||||
|
{{/unless}}
|
||||||
|
</div>
|
||||||
|
<div class="section" style="background-color: #e9e9e9;">
|
||||||
|
<div>
|
||||||
|
<div class="display1">Character Sheet Open Beta</div>
|
||||||
|
<h2 style="margin-bottom: 16px;">
|
||||||
|
Check out the example characters
|
||||||
|
</h2>
|
||||||
|
<div layout horizontal around-justified wrap>
|
||||||
|
<paper-shadow class="card characterCard ssArcher clickable"
|
||||||
|
z="2">
|
||||||
|
<div class="top subhead green white-text">
|
||||||
|
<div class="subhead" flex>
|
||||||
|
Starter Set Archer
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="bottom text">Lawful Good Human</div>
|
||||||
|
</paper-shadow>
|
||||||
|
<paper-shadow class="card characterCard ssWizard clickable"
|
||||||
|
z="2">
|
||||||
|
<div class="top subhead deep-purple white-text">
|
||||||
|
<div class="subhead" flex>
|
||||||
|
Starter Set Wizard
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="bottom text">Chaotic Good High Elf</div>
|
||||||
|
</paper-shadow>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="section white-text" style="background: #282828">
|
||||||
|
<div class="columns" layout horizontal around-justified wrap>
|
||||||
|
<div>
|
||||||
|
<h1>Check out the guide</h1>
|
||||||
|
<p>
|
||||||
|
Learn how your class gives you features, those features have effects,
|
||||||
|
and those effects determine your stats.
|
||||||
|
<paper-button class="guideButton">View Guide</paper-button>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<h1>
|
||||||
|
Discuss
|
||||||
|
</h1>
|
||||||
|
<p>
|
||||||
|
On the official subreddit
|
||||||
|
<paper-button class="redditButton">
|
||||||
|
<a href="http://www.reddit.com/r/dicecloud/">
|
||||||
|
/r/dicecloud
|
||||||
|
</a>
|
||||||
|
</paper-button>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<h1>
|
||||||
|
Get involved
|
||||||
|
</h1>
|
||||||
|
<p>Shape upcoming features and track bugs on the Dice Cloud Trello board
|
||||||
|
<paper-button class="trelloButton">
|
||||||
|
<a href="https://trello.com/b/94M0SCnq/dicecloud-roadmap">
|
||||||
|
Trello Roadmap
|
||||||
|
</a>
|
||||||
|
</paper-button>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
17
rpg-docs/client/views/home/intro/intro.js
Normal file
17
rpg-docs/client/views/home/intro/intro.js
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
Template.intro.events({
|
||||||
|
"tap .signInButton": function() {
|
||||||
|
Router.go("/sign-in");
|
||||||
|
},
|
||||||
|
"tap .signUpButton": function() {
|
||||||
|
Router.go("/sign-up");
|
||||||
|
},
|
||||||
|
"tap .ssArcher": function() {
|
||||||
|
Router.go("/character/yBWwt5XQTTHZiRQxq");
|
||||||
|
},
|
||||||
|
"tap .ssWizard": function() {
|
||||||
|
Router.go("/character/KxHKskm22fS2Xogah");
|
||||||
|
},
|
||||||
|
"tap .guideButton": function() {
|
||||||
|
Router.go("/guide");
|
||||||
|
},
|
||||||
|
});
|
||||||
33
rpg-docs/client/views/home/intro/intro.scss
Normal file
33
rpg-docs/client/views/home/intro/intro.scss
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
.intro {
|
||||||
|
.section {
|
||||||
|
width: 100%;
|
||||||
|
min-height: 200px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
flex-direction: column;
|
||||||
|
padding-top: 24px;
|
||||||
|
padding-bottom: 24px;
|
||||||
|
& > div, & > h2 {
|
||||||
|
padding: 32px;
|
||||||
|
.display1 {
|
||||||
|
margin-bottom: 16px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.columns > div{
|
||||||
|
max-width: 300px;
|
||||||
|
padding: 16px;
|
||||||
|
text-align: center;
|
||||||
|
paper-button {
|
||||||
|
color: #FF5252;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
paper-button {
|
||||||
|
min-width: 200px;
|
||||||
|
a {
|
||||||
|
color: inherit;
|
||||||
|
text-decoration: inherit;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user