Compare commits

...

12 Commits

Author SHA1 Message Date
Thaum Rystra
a41b267364 Use embedded property dialog in tree tab. Colors for creature properties 2020-05-25 19:36:14 +02:00
Thaum Rystra
dfb144b8dc Added color picking to library properties 2020-05-25 19:09:55 +02:00
Thaum Rystra
2859bf0e00 Added fade transition to library dialog 2020-05-25 18:41:22 +02:00
Thaum Rystra
469822d4d7 In organize mode, new library properties get placed under the selected node 2020-05-25 18:33:38 +02:00
Thaum Rystra
c7de96c8c3 Added "move" button to library property menu 2020-05-25 18:15:35 +02:00
Thaum Rystra
f7cbee27f9 Made selecting a property from a library use the mobile friendly library 2020-05-25 17:39:25 +02:00
Thaum Rystra
b61dd6e81a Added maximum length of ancestors array 2020-05-25 17:25:49 +02:00
Thaum Rystra
add0cac31d Added "duplicate" option to library properties 2020-05-25 17:23:36 +02:00
Thaum Rystra
e9c643699c Made tab swiping sync with the tab list 2020-05-25 17:07:38 +02:00
Thaum Rystra
3ec0f9500c Overhauled library UI to work on small screens 2020-05-25 16:43:28 +02:00
Thaum Rystra
a55c1382b1 Fixed skills not computing below zero 2020-05-24 04:30:14 +02:00
Thaum Rystra
7571806cd0 Made user profiles optional 2020-05-23 12:07:42 +02:00
35 changed files with 1043 additions and 633 deletions

View File

@@ -2,6 +2,7 @@ import { Meteor } from 'meteor/meteor';
import { Mongo } from 'meteor/mongo';
import { ValidatedMethod } from 'meteor/mdg:validated-method';
import SimpleSchema from 'simpl-schema';
import ColorSchema from '/imports/api/properties/subSchemas/ColorSchema.js';
import ChildSchema, { RefSchema } from '/imports/api/parenting/ChildSchema.js';
import { recomputeCreature } from '/imports/api/creature/computation/recomputeCreature.js';
import LibraryNodes from '/imports/api/library/LibraryNodes.js';
@@ -41,6 +42,7 @@ for (let key in propertySchemasIndex){
let schema = new SimpleSchema({});
schema.extend(propertySchemasIndex[key]);
schema.extend(CreaturePropertySchema);
schema.extend(ColorSchema);
schema.extend(ChildSchema);
schema.extend(SoftRemovableSchema);
CreatureProperties.attachSchema(schema, {
@@ -72,6 +74,7 @@ const insertProperty = new ValidatedMethod({
name: 'CreatureProperties.methods.insert',
validate: null,
run({creatureProperty}) {
delete creatureProperty._id;
assertPropertyEditPermission(creatureProperty, this.userId);
let _id = CreatureProperties.insert(creatureProperty);
let property = CreatureProperties.findOne(_id);
@@ -79,6 +82,23 @@ const insertProperty = new ValidatedMethod({
},
});
const duplicateProperty = new ValidatedMethod({
name: 'CreatureProperties.methods.duplicate',
validate: new SimpleSchema({
_id: {
type: String,
regEx: SimpleSchema.RegEx.Id,
}
}).validator(),
run({_id}) {
let creatureProperty = CreatureProperties.findOne(_id);
assertPropertyEditPermission(creatureProperty, this.userId);
delete creatureProperty._id;
CreatureProperties.insert(creatureProperty);
recomputeCreatures(creatureProperty);
},
});
const insertPropertyFromLibraryNode = new ValidatedMethod({
name: 'CreatureProperties.methods.insertPropertyFromLibraryNode',
validate: new SimpleSchema({
@@ -288,6 +308,7 @@ export default CreatureProperties;
export {
CreaturePropertySchema,
insertProperty,
duplicateProperty,
insertPropertyFromLibraryNode,
updateProperty,
damageProperty,

View File

@@ -51,7 +51,7 @@ function combineSkill(stat, aggregator, memo){
if (typeof profBonus !== 'number' && memo.statsByVariableName['level']){
let level = memo.statsByVariableName['level'].value;
profBonus = Math.floor(level / 4 + 1.75);
profBonus = Math.ceil(level / 4) + 1;
}
// Multiply the proficiency bonus by the actual proficiency
profBonus *= stat.proficiency;
@@ -60,7 +60,6 @@ function combineSkill(stat, aggregator, memo){
if (result < aggregator.min) result = aggregator.min;
if (result > aggregator.max) result = aggregator.max;
result = Math.floor(result);
if (aggregator.base > result) result = aggregator.base;
stat.value = result;
// Advantage/disadvantage
if (aggregator.advantage && !aggregator.disadvantage){

View File

@@ -2,6 +2,7 @@ import { Meteor } from 'meteor/meteor';
import { Mongo } from 'meteor/mongo';
import { ValidatedMethod } from 'meteor/mdg:validated-method';
import SimpleSchema from 'simpl-schema';
import ColorSchema from '/imports/api/properties/subSchemas/ColorSchema.js';
import ChildSchema from '/imports/api/parenting/ChildSchema.js';
import propertySchemasIndex from '/imports/api/properties/propertySchemasIndex.js';
import Libraries from '/imports/api/library/Libraries.js';
@@ -28,6 +29,7 @@ let LibraryNodeSchema = new SimpleSchema({
for (let key in propertySchemasIndex){
let schema = new SimpleSchema({});
schema.extend(LibraryNodeSchema);
schema.extend(ColorSchema);
schema.extend(propertySchemasIndex[key]);
schema.extend(ChildSchema);
schema.extend(SoftRemovableSchema);
@@ -52,11 +54,28 @@ const insertNode = new ValidatedMethod({
name: 'LibraryNodes.methods.insert',
validate: null,
run(libraryNode) {
delete libraryNode._id;
assertNodeEditPermission(libraryNode, this.userId);
return LibraryNodes.insert(libraryNode);
},
});
const duplicateNode = new ValidatedMethod({
name: 'LibraryNodes.methods.duplicate',
validate: new SimpleSchema({
_id: {
type: String,
regEx: SimpleSchema.RegEx.Id,
}
}).validator(),
run({_id}) {
let libraryNode = LibraryNodes.findOne(_id);
assertNodeEditPermission(libraryNode, this.userId);
delete libraryNode._id;
return LibraryNodes.insert(libraryNode);
},
})
const updateLibraryNode = new ValidatedMethod({
name: 'LibraryNodes.methods.update',
validate({_id, path}){
@@ -132,6 +151,7 @@ export default LibraryNodes;
export {
LibraryNodeSchema,
insertNode,
duplicateNode,
updateLibraryNode,
pullFromLibraryNode,
pushToLibraryNode,

View File

@@ -22,6 +22,7 @@ let ChildSchema = new SimpleSchema({
ancestors: {
type: Array,
defaultValue: [],
max: 100,
},
'ancestors.$': {
type: RefSchema,

View File

@@ -133,7 +133,7 @@ export function reorderDocs({collection, ancestorId}){
});
}
});
if (Meteor.isServer){
if (Meteor.isServer && bulkWrite.length){
collection.rawCollection().bulkWrite(
bulkWrite,
{ordered : false},

View File

@@ -80,6 +80,7 @@ const userSchema = new SimpleSchema({
profile: {
type: Object,
blackbox: true,
optional: true,
},
});

View File

@@ -1,193 +1,230 @@
<template lang="html">
<v-menu
:close-on-content-click="false"
<v-menu
v-model="opened"
:close-on-content-click="false"
transition="slide-y-transition"
lazy
v-model="opened"
>
<v-btn
slot="activator"
icon
>
<v-icon>format_paint</v-icon>
</v-btn>
<v-card class="overflow-hidden">
<v-card-text>
<v-item-group v-model="color" mandatory>
<v-layout row wrap>
<v-item
v-for="kebabColorOption in colors"
:key="kebabColorOption"
:value="kebabToCamelCase(kebabColorOption)"
>
<div
slot-scope="{ active, toggle }"
:class="[kebabColorOption, kebabShade]"
class="color-swatch d-flex align-center"
@click="toggle"
>
<v-scroll-y-transition>
<v-icon
v-if="active"
:class="{dark: isDark(kebabColorOption, kebabShade)}"
>check</v-icon>
</v-scroll-y-transition>
</div>
</v-item>
<div class="spacer" v-for="i in 8"/>
</v-layout>
</v-item-group>
<v-item-group class="mt-2" v-model="shade" mandatory>
<v-layout wrap>
<v-item
v-for="kebabShadeOption in shades"
:key="kebabShadeOption"
:value="kebabToCamelCase(kebabShadeOption)"
>
<div
slot-scope="{ active, toggle }"
:class="[kebabColor, kebabShadeOption]"
class="shade-swatch d-flex align-center"
@click="toggle"
>
<v-scroll-y-transition>
<v-icon
v-if="active"
:class="{dark: isDark(kebabColor, kebabShadeOption)}"
>check</v-icon>
</v-scroll-y-transition>
</div>
</v-item>
<div class="spacer" v-for="i in 8"/>
</v-layout>
</v-item-group>
</v-card-text>
<v-card-actions>
<v-spacer/>
<v-btn flat @click="opened = false">Done</v-btn>
</v-card-actions>
</v-card>
lazy
left
>
<v-btn
slot="activator"
icon
>
<v-icon>format_paint</v-icon>
</v-btn>
<v-card class="overflow-hidden">
<v-card-text>
<v-layout
row
wrap
>
<div
v-for="colorOption in colors"
:key="colorOption"
:class="[colorOption, shade]"
class="color-swatch d-flex align-center"
@click="color = colorOption"
>
<v-scroll-y-transition>
<v-icon
v-if="kebabColor === colorOption"
:class="{dark: isDark(colorOption, shade)}"
>
check
</v-icon>
</v-scroll-y-transition>
</div>
<div
v-for="i in 8"
:key="i"
class="spacer"
/>
</v-layout>
<v-fade-transition>
<v-layout
v-show="color"
wrap
class="mt-2"
>
<div
v-for="shadeOption in shades"
:key="shadeOption"
:class="[kebabColor, shadeOption]"
class="shade-swatch d-flex align-center"
@click="shade = shadeOption"
>
<v-scroll-y-transition>
<v-icon
v-if="kebabShade === shadeOption"
:class="{dark: isDark(color, shade)}"
>
check
</v-icon>
</v-scroll-y-transition>
</div>
<div
v-for="i in 8"
:key="i"
class="spacer"
/>
</v-layout>
</v-fade-transition>
</v-card-text>
<v-card-actions>
<v-btn
flat
@click="$emit('input')"
>
Clear
</v-btn>
<v-spacer />
<v-btn
flat
@click="opened = false"
>
Done
</v-btn>
</v-card-actions>
</v-card>
</v-menu>
</template>
<script>
import isDarkColor from '/imports/ui/utility/isDarkColor.js';
import vuetifyColors from 'vuetify/es5/util/colors';
import { kebabToCamelCase, camelToKebabCase } from '/imports/ui/utility/swapCase.js';
import isDarkColor from '/imports/ui/utility/isDarkColor.js';
import vuetifyColors from 'vuetify/es5/util/colors';
import { kebabToCamelCase, camelToKebabCase } from '/imports/ui/utility/swapCase.js';
function colorToHex(color, shade = 'base'){
color = kebabToCamelCase(color);
shade = kebabToCamelCase(shade);
return vuetifyColors[color][shade];
};
function colorToHex(color, shade = 'base'){
if (!color) return;
color = kebabToCamelCase(color);
shade = kebabToCamelCase(shade);
return vuetifyColors[color] && vuetifyColors[color][shade];
}
// Create an index of hex colors and what color/shade combination makes them
let colorIndex = {};
for (let color in vuetifyColors){
for (let shade in vuetifyColors[color]){
colorIndex[vuetifyColors[color][shade]] = {color, shade};
}
}
function hexToColor(hex){
return colorIndex[hex.toLowerCase()];
};
// Create an index of hex colors and what color/shade combination makes them
let colorIndex = {};
for (let color in vuetifyColors){
color = kebabToCamelCase(color);
for (let shade in vuetifyColors[color]){
shade = kebabToCamelCase(shade);
colorIndex[vuetifyColors[color][shade]] = {color, shade};
}
}
function hexToColor(hex){
if (!hex) return undefined;
return colorIndex[hex.toLowerCase()];
}
export default {
props: {
value: String, //hex string
},
data(){ return {
colors: [
'red',
'pink',
'purple',
'deep-purple',
'indigo',
'blue',
'light-blue',
'cyan',
'teal',
'green',
'light-green',
'lime',
'yellow',
'amber',
'orange',
'deep-orange',
'brown',
'grey',
],
shades: [
'lighten-4',
'lighten-3',
'lighten-2',
'lighten-1',
'base',
'darken-1',
'darken-2',
'darken-3',
'darken-4',
],
opened: false,
}},
methods: {
isDark(kebabColor, kebabShade){
color = colorToHex(kebabColor, kebabShade);
return isDarkColor(color);
},
kebabToCamelCase,
},
computed: {
combination (){
return hexToColor(this.value) || {};
},
color: {
get(){
return this.combination.color;
},
set(newColor){
this.$emit('input', colorToHex(newColor, this.shade));
},
},
shade: {
get(){
return this.combination.shade;
},
set(newShade){
this.$emit('input', colorToHex(this.color, newShade));
},
},
kebabColor(){
return camelToKebabCase(this.color);
},
kebabShade(){
return camelToKebabCase(this.shade);
},
},
};
export default {
props: {
//hex string
value: {
type: String,
default: undefined,
},
},
data(){ return {
colors: [
'red',
'pink',
'purple',
'deep-purple',
'indigo',
'blue',
'light-blue',
'cyan',
'teal',
'green',
'light-green',
'lime',
'yellow',
'amber',
'orange',
'deep-orange',
'brown',
'grey',
],
shades: [
'lighten-4',
'lighten-3',
'lighten-2',
'lighten-1',
'base',
'darken-1',
'darken-2',
'darken-3',
'darken-4',
],
opened: false,
}},
computed: {
combination (){
if (!this.value) return;
return hexToColor(this.value) || {};
},
color: {
get(){
return this.combination && this.combination.color;
},
set(newColor){
this.$emit('input', colorToHex(newColor, this.shade));
},
},
shade: {
get(){
return this.combination && this.combination.shade;
},
set(newShade){
this.$emit('input', colorToHex(this.color, newShade));
},
},
kebabColor(){
return camelToKebabCase(this.color);
},
kebabShade(){
return camelToKebabCase(this.shade);
},
},
methods: {
isDark(kebabColor, kebabShade){
let color = colorToHex(kebabColor, kebabShade);
return isDarkColor(color);
},
kebabToCamelCase,
},
};
</script>
<style lang="css" scoped>
.color-swatch, .shade-swatch {
height: 30px;
width: 30px;
flex-grow: 1;
}
.v-icon {
height: 30px;
}
.v-icon {
color: black;
}
.dark.v-icon {
color: white;
}
.layout {
max-width: 270px;
}
.spacer {
width: 30px;
height: 0;
flex-grow: 1;
}
.color-swatch, .shade-swatch {
height: 30px;
width: 30px;
flex-grow: 1;
cursor: pointer;
transition: all 0.2s linear;
}
.color-swatch:hover{
z-index: 1;
transform: scale(1.1);
box-shadow: 0px 2px 1px -1px rgba(0,0,0,0.2),
0px 1px 1px 0px rgba(0,0,0,0.14),
0px 1px 3px 0px rgba(0,0,0,0.12);
}
.v-icon {
height: 30px;
}
.v-icon {
color: black;
}
.dark.v-icon {
color: white;
}
.layout {
max-width: 270px;
}
.spacer {
width: 30px;
height: 0;
flex-grow: 1;
}
</style>

View File

@@ -0,0 +1,43 @@
<template lang="html">
<div
class="layout row"
style="height: 100%;"
>
<div
class="layout column justify-start"
:style="computedTreeStyle"
>
<slot name="tree" />
</div>
<template v-if="$vuetify.breakpoint.mdAndUp">
<v-divider vertical />
<div
class="flex layout column"
style="background-color: inherit; overflow: hidden;"
data-id="selected-node-card"
>
<slot name="detail" />
</div>
</template>
</div>
</template>
<script>
export default {
computed:{
computedTreeStyle(){
if (this.$vuetify.breakpoint.smAndDown) return;
let style = 'flex-shrink: 0; flex-grow: 0; ';
if (this.$vuetify.breakpoint.xlOnly){
style += 'width: 400px;'
} else {
style += 'width: 320px;'
}
return style;
},
}
}
</script>
<style lang="css" scoped>
</style>

View File

@@ -21,7 +21,7 @@
props: {
autoGrow: {
type: Boolean,
default: true,
default: false,
},
},
};

View File

@@ -0,0 +1,156 @@
<template lang="html">
<v-toolbar
:color="color || 'secondary'"
:dark="isDark"
:flat="flat"
>
<property-icon
:type="model && model.type"
class="mr-2"
/>
<v-toolbar-title v-if="model">
{{ model.name || getPropertyName(model.type) }}
</v-toolbar-title>
<v-spacer />
<v-slide-y-transition
hide-on-leave
>
<v-layout
v-if="editing && model"
key="edit-buttons"
>
<v-spacer />
<color-picker
v-if="$listeners && $listeners['color-changed']"
:value="model.color"
@input="colorChanged"
/>
<v-menu
v-if="$listeners && (
$listeners.move ||
$listeners.duplicate ||
$listeners.remove
)"
bottom
left
transition="slide-y-transition"
>
<template #activator="{ on }">
<v-btn
icon
data-id="property-toolbar-menu-button"
v-on="on"
>
<v-icon>more_vert</v-icon>
</v-btn>
</template>
<v-list>
<v-list-tile
v-if="$listeners && $listeners.duplicate"
@click="$emit('duplicate')"
>
<v-list-tile-content>
<v-list-tile-title>
Duplicate
</v-list-tile-title>
</v-list-tile-content>
<v-list-tile-action>
<v-icon>file_copy</v-icon>
</v-list-tile-action>
</v-list-tile>
<v-list-tile
v-if="$listeners && $listeners.move"
@click="$emit('move')"
>
<v-list-tile-content>
<v-list-tile-title>
Move
</v-list-tile-title>
</v-list-tile-content>
<v-list-tile-action>
<v-icon>send</v-icon>
</v-list-tile-action>
</v-list-tile>
<v-list-tile
v-if="$listeners && $listeners.remove"
@click="$emit('remove')"
>
<v-list-tile-content>
<v-list-tile-title>
Delete
</v-list-tile-title>
</v-list-tile-content>
<v-list-tile-action>
<v-icon>delete</v-icon>
</v-list-tile-action>
</v-list-tile>
</v-list>
</v-menu>
</v-layout>
<v-layout
v-else
key="blank"
/>
</v-slide-y-transition>
<v-btn
icon
@click="$emit('toggle-editing')"
>
<v-slide-y-transition
hide-on-leave
>
<v-icon
v-if="editing"
key="doneIcon"
>
done
</v-icon>
<v-icon
v-else
key="createIcon"
>
create
</v-icon>
</v-slide-y-transition>
</v-btn>
</v-toolbar>
</template>
<script>
import isDarkColor from '/imports/ui/utility/isDarkColor.js';
import PropertyIcon from '/imports/ui/properties/shared/PropertyIcon.vue';
import { getPropertyName } from '/imports/constants/PROPERTIES.js';
import ColorPicker from '/imports/ui/components/ColorPicker.vue';
export default {
components: {
PropertyIcon,
ColorPicker,
},
props: {
model: {
type: Object,
default: undefined,
},
flat: Boolean,
editing: Boolean,
},
computed: {
isDark(){
return isDarkColor(this.color);
},
color(){
return this.model && this.model.color || this.$vuetify.theme.secondary;
}
},
methods: {
colorChanged(value){
this.$emit('color-changed', value);
},
getPropertyName,
}
}
</script>
<style lang="css" scoped>
</style>

View File

@@ -46,7 +46,7 @@
v-show="showExpanded"
class="pl-3"
>
<v-fade-transition leave-absolute>
<v-fade-transition hide-on-leave>
<tree-node-list
v-if="showExpanded"
:node="node"

View File

@@ -35,7 +35,7 @@
class="fill-height"
>
<v-tabs-items
v-model="tabs"
v-model="activeTab"
>
<v-tab-item>
<stats-tab :creature-id="creatureId" />
@@ -103,6 +103,16 @@
'creature.name'(value){
this.$store.commit('setPageTitle', value || 'Character Sheet');
},
},
computed: {
activeTab: {
get(){
return this.tabs;
},
set(newTab){
this.$emit('update:tabs', newTab);
},
},
},
meteor: {
$subscribe: {

View File

@@ -2,6 +2,7 @@
<v-tabs
v-if="creature"
slot="extension"
color="secondary"
:value="value"
centered
grow

View File

@@ -1,106 +1,60 @@
<template lang="html">
<div
class="tree-tab pa-4"
class="tree-tab pa-4 layout column align-center"
style="height: calc(100vh - 96px); display: flex;"
>
<v-card
class="layout row"
style="height: 100%;"
style="height: 100%; width: 100%; max-width: 1800px;"
data-id="creature-tree-card"
>
<div
class="layout column justify-start"
:style="
$vuetify.breakpoint.mdAndUp &&
'width: 320px; flex-shrink: 0; flex-grow: 0;'
"
>
<v-toolbar
flat
dense
>
<v-spacer />
<v-switch
v-if="context.editPermission !== false"
v-model="organize"
label="Organize"
class="mx-3"
:disabled="organizeDisabled"
style="flex-grow: 0; height: 32px;"
/>
<v-combobox
ref="searchBox"
slot="extension"
v-model="filterString"
:items="filterOptions"
prepend-inner-icon="search"
class="mx-4"
hide-no-data
hide-selected
multiple
clearable
small-chips
deletable-chips
/>
</v-toolbar>
<creature-properties-tree
class="pt-2 flex"
style="overflow-y: auto;"
:root="{collection: 'creatures', id: creatureId}"
:organize="organize"
:selected-node-id="selected"
:filter="filter"
@selected="clickNode"
/>
</div>
<template v-if="$vuetify.breakpoint.mdAndUp">
<v-divider vertical />
<div
class="flex layout column"
style="background-color: inherit; overflow: hidden;"
data-id="selected-node-card"
>
<tree-detail-layout>
<template slot="tree">
<v-toolbar
dense
flat
extended
dense
>
<v-fade-transition mode="out-in">
<div
:key="selectedProperty && selectedProperty._id"
class="title"
>
<property-icon
:key="selectedProperty && selectedProperty._id"
:type="selectedProperty && selectedProperty.type"
class="mr-2"
/>
{{ getPropertyName(selectedProperty && selectedProperty.type) }}
</div>
</v-fade-transition>
<v-spacer />
<v-btn
v-if="selectedProperty"
flat
icon
@click="editCreatureProperty"
>
<v-icon>create</v-icon>
</v-btn>
<v-switch
v-if="context.editPermission !== false"
v-model="organize"
label="Organize"
class="mx-3"
:disabled="organizeDisabled"
style="flex-grow: 0; height: 32px;"
/>
<v-combobox
ref="searchBox"
slot="extension"
v-model="filterString"
:items="filterOptions"
prepend-inner-icon="search"
class="mx-4"
hide-no-data
hide-selected
multiple
clearable
small-chips
deletable-chips
/>
</v-toolbar>
<v-card-text
class="flex"
style="overflow-y: auto"
>
<v-fade-transition mode="out-in">
<property-viewer
:key="selectedProperty && selectedProperty._id"
:model="selectedProperty"
/>
</v-fade-transition>
</v-card-text>
</div>
</template>
<creature-properties-tree
class="pt-2 flex"
style="overflow-y: auto;"
:root="{collection: 'creatures', id: creatureId}"
:organize="organize"
:selected-node-id="selected"
:filter="filter"
@selected="clickNode"
/>
</template>
<template slot="detail">
<creature-property-dialog
embedded
:_id="selected"
@removed="selected = undefined"
/>
</template>
</tree-detail-layout>
</v-card>
<v-speed-dial
v-model="fab"
@@ -136,22 +90,23 @@
</template>
<script>
import TreeDetailLayout from '/imports/ui/components/TreeDetailLayout.vue';
import CreaturePropertiesTree from '/imports/ui/creature/creatureProperties/CreaturePropertiesTree.vue';
import CreaturePropertyDialog from '/imports/ui/creature/creatureProperties/CreaturePropertyDialog.vue';
import LabeledFab from '/imports/ui/components/LabeledFab.vue';
import CreatureProperties, {
insertProperty,
insertPropertyFromLibraryNode
} from '/imports/api/creature/CreatureProperties.js';
import PropertyViewer from '/imports/ui/properties/shared/PropertyViewer.vue';
import { setDocToLastOrder } from '/imports/api/parenting/order.js';
import PropertyIcon from '/imports/ui/properties/shared/PropertyIcon.vue';
import { getPropertyName } from '/imports/constants/PROPERTIES.js';
import LabeledFab from '/imports/ui/components/LabeledFab.vue';
export default {
components: {
TreeDetailLayout,
CreaturePropertiesTree,
PropertyViewer,
PropertyIcon,
CreaturePropertyDialog,
LabeledFab,
},
inject: {

View File

@@ -1,79 +1,40 @@
<template lang="html">
<dialog-base>
<template slot="toolbar">
<property-icon
:type="model.type"
class="mr-2"
<template #replace-toolbar="{flat}">
<property-toolbar
:model="model"
:editing="editing"
:flat="flat"
@duplicate="duplicate"
@remove="remove"
@toggle-editing="editing = !editing"
@color-changed="value => change({path: ['color'], value})"
/>
<v-toolbar-title>
{{ model.name || getPropertyName(model.type) }}
</v-toolbar-title>
<v-spacer />
<v-menu
v-if="editing"
bottom
left
transition="slide-y-transition"
>
<template #activator="{ on }">
<v-btn
icon
v-on="on"
>
<v-icon>more_vert</v-icon>
</v-btn>
</template>
<v-list>
<v-list-tile @click="remove">
<v-list-tile-title>
Delete <v-icon>delete</v-icon>
</v-list-tile-title>
</v-list-tile>
</v-list>
</v-menu>
<v-btn
icon
@click="editing = !editing"
>
<v-slide-y-transition
leave-absolute
mode="out-in"
>
<v-icon
v-if="editing"
key="doneIcon"
>
done
</v-icon>
<v-icon
v-else
key="createIcon"
>
create
</v-icon>
</v-slide-y-transition>
</v-btn>
</template>
<template v-if="model">
<component
:is="model.type + 'Form'"
v-if="editing"
class="creature-property-form"
:model="model"
@change="change"
@push="push"
@pull="pull"
/>
<component
:is="model.type + 'Viewer'"
v-else-if="!editing && $options.components[model.type + 'Viewer']"
class="creature-property-viewer"
:model="model"
/>
<p v-else>
This property can't be viewed yet.
</p>
<template v-if="!editing">
<v-fade-transition
mode="out-in"
>
<component
:is="model.type + 'Form'"
v-if="editing"
class="creature-property-form"
:model="model"
@change="change"
@push="push"
@pull="pull"
/>
<component
:is="model.type + 'Viewer'"
v-else-if="!editing && $options.components[model.type + 'Viewer']"
class="creature-property-viewer"
:model="model"
/>
<p v-else>
This property can't be viewed yet.
</p>
</v-fade-transition>
<template v-if="!editing && !embedded">
<v-divider />
<creature-properties-tree
v-if="!editing"
@@ -83,6 +44,7 @@
</template>
</template>
<div
v-if="!embedded"
slot="actions"
class="layout row justify-end"
>
@@ -100,11 +62,13 @@
import CreatureProperties, {
updateProperty,
damageProperty,
duplicateProperty,
pushToProperty,
pullFromProperty,
softRemoveProperty,
} from '/imports/api/creature/CreatureProperties.js';
import Creatures from '/imports/api/creature/Creatures.js';
import PropertyToolbar from '/imports/ui/components/propertyToolbar.vue';
import DialogBase from '/imports/ui/dialogStack/DialogBase.vue';
import { getPropertyName } from '/imports/constants/PROPERTIES.js';
import PropertyIcon from '/imports/ui/properties/shared/PropertyIcon.vue';
@@ -130,10 +94,12 @@ export default {
...viewerIndex,
PropertyIcon,
DialogBase,
PropertyToolbar,
CreaturePropertiesTree,
},
props: {
_id: String,
embedded: Boolean, // This dialog is embedded in a page
startInEditTab: Boolean,
},
data(){ return {
@@ -169,8 +135,26 @@ export default {
},
methods: {
getPropertyName,
duplicate(){
duplicateProperty.call({_id: this._id}, (error) => {
if (error) {
console.error(error);
}
if (this.embedded){
this.$emit('duplicated');
} else {
this.$store.dispatch('popDialogStack');
}
});
},
change({path, value, ack}){
updateProperty.call({_id: this._id, path, value}, (error) =>{
if (error) console.warn(error);
ack && ack(error && error.reason || error);
});
},
damage({operation, value, ack}){
damageProperty.call({_id: this._id, operation, value}, (error) =>{
if (error) console.warn(error);
ack && ack(error && error.reason || error);
});

View File

@@ -6,6 +6,7 @@
<library-and-node
slot="unwrapped-content"
style="height: 100%;"
selection
@selected="val => node = val"
/>
<template slot="actions">

View File

@@ -1,64 +1,72 @@
<template>
<v-layout column style="height: 100%;">
<v-toolbar :color="color || 'secondary'" dark class="base-dialog-toolbar" :flat="!offsetTop">
<v-btn icon flat @click="back">
<v-icon>arrow_back</v-icon>
</v-btn>
<slot name="toolbar"/>
<template v-if="$slots.edit">
<v-spacer/>
<v-btn icon flat @click="$emit('remove')" v-if="isEditing">
<v-icon>delete</v-icon>
</v-btn>
<v-btn icon flat @click="isEditing = !isEditing">
<v-icon>{{isEditing ? 'check' : 'create'}}</v-icon>
</v-btn>
</template>
</v-toolbar>
<template v-if="breadcrumbs">
<v-card-text>
example > bread > crumb
</v-card-text>
</template>
<div
v-if="$slots['unwrapped-content']"
class="unwrapped-content"
>
<slot name="unwrapped-content"/>
</div>
<v-card-text
v-if="!$slots['unwrapped-content']"
id="base-dialog-body"
v-scroll:#base-dialog-body="onScroll"
>
<v-tabs-items :value="isEditing ? 1 : 0" touchless>
<v-tab-item>
<slot/>
</v-tab-item>
<v-tab-item lazy>
<slot name="edit"/>
</v-tab-item>
</v-tabs-items>
</v-card-text>
<v-card-actions>
<slot name="actions"/>
</v-card-actions>
</v-layout>
<v-layout
column
style="height: 100%;"
>
<slot
name="replace-toolbar"
:flat="!offsetTop"
>
<v-toolbar
:color="computedColor"
:dark="isDark"
class="base-dialog-toolbar"
:flat="!offsetTop"
>
<v-btn
icon
flat
@click="back"
>
<v-icon>arrow_back</v-icon>
</v-btn>
<slot name="toolbar" />
</v-toolbar>
</slot>
<div
v-if="$slots['unwrapped-content']"
class="unwrapped-content"
>
<slot name="unwrapped-content" />
</div>
<v-card-text
v-if="!$slots['unwrapped-content']"
id="base-dialog-body"
v-scroll:#base-dialog-body="onScroll"
>
<slot />
</v-card-text>
<v-card-actions v-if="$slots.actions">
<slot name="actions" />
</v-card-actions>
</v-layout>
</template>
<script>
import store from "/imports/ui/vuexStore.js";
import isDarkColor from '/imports/ui/utility/isDarkColor.js';
export default {
props: {
color: String,
breadcrumbs: Object,
overrideBackButton: Function,
color: {
type: String,
default: undefined,
},
overrideBackButton: {
type: Function,
default: undefined,
},
},
data(){ return {
offsetTop: 0,
isEditing: false,
}},
computed: {
isDark(){
return isDarkColor(this.computedColor);
},
computedColor(){
return this.color || this.$vuetify.theme.secondary;
}
},
methods: {
onScroll(e){
this.offsetTop = e.target.scrollTop
@@ -71,7 +79,7 @@
}
},
close(){
store.dispatch("popDialogStack");
this.$store.dispatch('popDialogStack');
},
},
}

View File

@@ -7,7 +7,8 @@ import InviteDialog from '/imports/ui/user/InviteDialog.vue';
import LibraryCreationDialog from '/imports/ui/library/LibraryCreationDialog.vue';
import LibraryEditDialog from '/imports/ui/library/LibraryEditDialog.vue';
import LibraryNodeCreationDialog from '/imports/ui/library/LibraryNodeCreationDialog.vue';
import LibraryNodeEditDialog from '/imports/ui/library/LibraryNodeEditDialog.vue';
import LibraryNodeDialog from '/imports/ui/library/LibraryNodeDialog.vue';
import MoveLibraryNodeDialog from '/imports/ui/library/MoveLibraryNodeDialog.vue'
import ShareDialog from '/imports/ui/sharing/ShareDialog.vue';
import TierTooLowDialog from '/imports/ui/user/TierTooLowDialog.vue';
import UsernameDialog from '/imports/ui/user/UsernameDialog.vue';
@@ -23,7 +24,8 @@ export default {
LibraryCreationDialog,
LibraryEditDialog,
LibraryNodeCreationDialog,
LibraryNodeEditDialog,
LibraryNodeDialog,
MoveLibraryNodeDialog,
ShareDialog,
TierTooLowDialog,
UsernameDialog,

View File

@@ -54,20 +54,10 @@
</v-fade-transition>
</v-toolbar>
<v-content>
<v-alert
v-if="$route.path !== '/countdown'"
icon="priority_high"
type="error"
dismissible
:value="true"
>
This version of DiceCloud is in beta. Some data stored here may be destroyed by
future updates.
</v-alert>
<v-fade-transition
mode="out-in"
>
<router-view :tabs="tabs" />
<router-view :tabs.sync="tabs" />
</v-fade-transition>
</v-content>

View File

@@ -1,5 +1,15 @@
<template>
<div class="sidebar">
<v-alert
v-if="$route.path !== '/countdown'"
icon="priority_high"
type="error"
dismissible
:value="true"
>
This version of DiceCloud is in beta. Some data stored here may be destroyed by
future updates.
</v-alert>
<v-layout
v-if="!signedIn"
row

View File

@@ -1,20 +1,20 @@
<template lang="html">
<div
class="layout row"
style="background-color: inherit;"
>
<tree-detail-layout>
<div
slot="tree"
class="layout column"
style="
background-color: inherit;
width: initial;
max-width: 100%;
min-width: 320px;
"
background-color: inherit;
width: initial;
max-width: 100%;
min-width: 320px;
height: 100%;
"
>
<v-toolbar
dense
flat
:color="selectedNode && selectedNode.color || 'secondary'"
:dark="isDarkColor(selectedNode && selectedNode.color || $vuetify.theme.secondary)"
>
<v-spacer />
<v-switch
@@ -28,91 +28,97 @@
edit-mode
:organize-mode="organize"
:selected-node-id="selected"
@selected="e => selected = e"
style="overflow-y: auto;"
@selected="clickNode"
/>
</div>
<v-divider vertical />
<div
style="width: 100%; background-color: inherit;"
slot="detail"
data-id="selected-node-card"
>
<v-toolbar
dense
flat
>
<property-icon
:type="selectedNode && selectedNode.type"
class="mr-2"
/>
<div class="title">
{{ getPropertyName(selectedNode && selectedNode.type) }}
</div>
<v-spacer />
<v-btn
v-if="selectedNode"
flat
icon
@click="editLibraryNode"
>
<v-icon>create</v-icon>
</v-btn>
</v-toolbar>
<v-card-text style="overflow-y: auto;">
<property-viewer :model="selectedNode" />
</v-card-text>
<library-node-dialog
:_id="selected"
embedded
@removed="selected = undefined"
/>
</div>
</div>
</tree-detail-layout>
</template>
<script>
import TreeDetailLayout from '/imports/ui/components/TreeDetailLayout.vue';
import LibraryBrowser from '/imports/ui/library/LibraryBrowser.vue';
import PropertyViewer from '/imports/ui/properties/shared/PropertyViewer.vue';
import LibraryNodeDialog from '/imports/ui/library/LibraryNodeDialog.vue';
import LibraryNodes from '/imports/api/library/LibraryNodes.js';
import Libraries from '/imports/api/library/Libraries.js';
import PropertyIcon from '/imports/ui/properties/shared/PropertyIcon.vue';
import { getPropertyName } from '/imports/constants/PROPERTIES.js';
import isDarkColor from '/imports/ui/utility/isDarkColor.js';
export default {
components: {
LibraryBrowser,
PropertyViewer,
PropertyIcon,
},
data(){ return {
organize: false,
selected: undefined,
};},
watch:{
selectedNode(val){
this.$emit('selected', val)
},
},
methods: {
editLibraryNode(){
this.$store.commit('pushDialogStack', {
component: 'library-node-edit-dialog',
elementId: 'selected-node-card',
data: {_id: this.selected},
});
},
getPropertyName,
},
meteor: {
$subscribe: {
'libraries': [],
},
libraries(){
return Libraries.find({}, {
sort: {name: 1}
}).fetch();
},
selectedNode(){
return LibraryNodes.findOne({
_id: this.selected,
removed: {$ne: true}
});
}
}
components: {
TreeDetailLayout,
LibraryBrowser,
LibraryNodeDialog,
},
props: {
selection: Boolean,
},
data(){ return {
organize: false,
selected: undefined,
};},
watch:{
selectedNode(val){
this.$emit('selected', val)
},
},
methods: {
editLibraryNode(){
this.$store.commit('pushDialogStack', {
component: 'library-node-edit-dialog',
elementId: 'selected-node-card',
data: {_id: this.selected},
});
},
clickNode(id){
if (this.$vuetify.breakpoint.mdAndUp){
this.selected = id;
} else {
this.$store.commit('pushDialogStack', {
component: 'library-node-dialog',
elementId: `tree-node-${id}`,
data: {
_id: id,
selection: this.selection,
},
callback: result => {
console.log(result)
if (result){
this.selected = id;
}
},
});
}
},
getPropertyName,
isDarkColor,
},
meteor: {
$subscribe: {
'libraries': [],
},
libraries(){
return Libraries.find({}, {
sort: {name: 1}
}).fetch();
},
selectedNode(){
return LibraryNodes.findOne({
_id: this.selected,
removed: {$ne: true}
});
},
}
};
</script>

View File

@@ -76,6 +76,7 @@ import LibraryNodes, { insertNode } from '/imports/api/library/LibraryNodes.js';
import Libraries, { insertLibrary } from '/imports/api/library/Libraries.js';
import { getUserTier } from '/imports/api/users/patreon/tiers.js';
import { assertEditPermission } from '/imports/api/sharing/sharingPermissions.js';
import { getAncestry } from '/imports/api/parenting/parenting.js';
export default {
components: {
@@ -139,13 +140,20 @@ export default {
},
insertLibraryNode(libraryId){
if (this.paidBenefits){
let parentRef;
if (this.organizeMode && this.selectedNodeId){
parentRef = {collection: 'libraryNodes', id: this.selectedNodeId}
} else {
parentRef = {collection: 'libraries', id: libraryId};
}
let {ancestors} = getAncestry({parentRef});
this.$store.commit('pushDialogStack', {
component: 'library-node-creation-dialog',
elementId: `insert-node-${libraryId}`,
callback(libraryNode){
if (!libraryNode) return;
libraryNode.parent = {collection: 'libraries', id: libraryId};
libraryNode.ancestors = [ {collection: 'libraries', id: libraryId}];
libraryNode.parent = parentRef;
libraryNode.ancestors = ancestors;
setDocToLastOrder({collection: LibraryNodes, doc: libraryNode});
let libraryNodeId = insertNode.call(libraryNode);
return `tree-node-${libraryNodeId}`;

View File

@@ -0,0 +1,209 @@
<template lang="html">
<dialog-base>
<template #replace-toolbar="{flat}">
<property-toolbar
:model="model"
:editing="editing"
:flat="flat"
@duplicate="duplicate"
@move="move"
@remove="remove"
@toggle-editing="editing = !editing"
@color-changed="value => change({path: ['color'], value})"
/>
</template>
<template v-if="model">
<v-fade-transition
mode="out-in"
>
<component
:is="model.type + 'Form'"
v-if="editing"
class="library-node-form"
:model="model"
@change="change"
@push="push"
@pull="pull"
/>
<component
:is="model.type + 'Viewer'"
v-else-if="!editing && $options.components[model.type + 'Viewer']"
class="creature-property-viewer"
:model="model"
/>
<p v-else>
This property can't be viewed yet.
</p>
</v-fade-transition>
</template>
<div
v-if="!embedded"
slot="actions"
class="layout row justify-end"
>
<template v-if="selection">
<v-btn
flat
@click="$store.dispatch('popDialogStack', false)"
>
Cancel
</v-btn>
<v-spacer />
<v-btn
flat
@click="$store.dispatch('popDialogStack', true)"
>
Select
</v-btn>
</template>
<v-btn
v-else
flat
@click="$store.dispatch('popDialogStack')"
>
Done
</v-btn>
</div>
</dialog-base>
</template>
<script>
import LibraryNodes, {
duplicateNode,
updateLibraryNode,
pushToLibraryNode,
pullFromLibraryNode,
softRemoveLibraryNode,
} from '/imports/api/library/LibraryNodes.js';
import DialogBase from '/imports/ui/dialogStack/DialogBase.vue';
import PropertyToolbar from '/imports/ui/components/propertyToolbar.vue';
import { getPropertyName } from '/imports/constants/PROPERTIES.js';
import PropertyIcon from '/imports/ui/properties/shared/PropertyIcon.vue';
import propertyFormIndex from '/imports/ui/properties/forms/shared/propertyFormIndex.js';
import propertyViewerIndex from '/imports/ui/properties/viewers/shared/propertyViewerIndex.js';
import { get } from 'lodash';
import { assertDocEditPermission } from '/imports/api/sharing/sharingPermissions.js';
import { organizeDoc } from '/imports/api/parenting/organizeMethods.js';
let formIndex = {};
for (let key in propertyFormIndex){
formIndex[key + 'Form'] = propertyFormIndex[key];
}
let viewerIndex = {};
for (let key in propertyViewerIndex){
formIndex[key + 'Viewer'] = propertyViewerIndex[key];
}
export default {
components: {
PropertyToolbar,
PropertyIcon,
DialogBase,
...formIndex,
...viewerIndex,
},
props: {
_id: String,
startInEditTab: Boolean,
embedded: Boolean, // This dialog is embedded in a page
selection: Boolean, // This dialog is being used to select a node
},
reactiveProvide: {
name: 'context',
include: ['editPermission'],
},
data(){return {
editing: !!this.startInEditTab,
}},
meteor: {
model(){
return LibraryNodes.findOne(this._id);
},
editPermission(){
try {
assertDocEditPermission(this.model, Meteor.userId());
return true;
} catch (e) {
return false;
}
},
},
methods: {
getPropertyName,
duplicate(){
duplicateNode.call({_id: this._id}, (error) => {
console.error(error);
if (this.embedded){
this.$emit('duplicated');
} else {
this.$store.dispatch('popDialogStack');
}
});
},
move(){
let that = this;
this.$store.commit('pushDialogStack', {
component: 'move-library-node-dialog',
elementId: 'property-toolbar-menu-button',
callback(parentId){
if (!parentId) return;
organizeDoc.call({
docRef: {
collection: 'libraryNodes',
id: that._id,
},
parentRef: {
collection: 'libraryNodes',
id: parentId
},
order: -0.5
}, (error) => {
if (error) console.error(error);
});
}
});
},
change({path, value, ack}){
updateLibraryNode.call({_id: this._id, path, value}, (error) =>{
if (ack){
ack(error && error.reason || error);
} else if (error){
console.error(error);
}
});
},
push({path, value, ack}){
pushToLibraryNode.call({_id: this._id, path, value}, (error) =>{
if (ack){
ack(error && error.reason || error);
} else if (error){
console.error(error);
}
});
},
pull({path, ack}){
let itemId = get(this.model, path)._id;
path.pop();
pullFromLibraryNode.call({_id: this._id, path, itemId}, (error) =>{
if (ack){
ack(error && error.reason || error);
} else if (error){
console.error(error);
}
});
},
remove(){
softRemoveLibraryNode.call({_id: this._id});
if (this.embedded){
this.$emit('removed');
} else {
this.$store.dispatch('popDialogStack');
}
},
}
};
</script>
<style lang="css" scoped>
</style>

View File

@@ -1,114 +0,0 @@
<template lang="html">
<dialog-base>
<template slot="toolbar">
<property-icon
:type="model.type"
class="mr-2"
/>
<v-toolbar-title>
{{ getPropertyName(model.type) }}
</v-toolbar-title>
<v-spacer />
<v-btn
icon
flat
@click="remove"
>
<v-icon>delete</v-icon>
</v-btn>
</template>
<component
:is="model.type"
v-if="model"
class="library-node-form"
:model="model"
@change="change"
@push="push"
@pull="pull"
/>
<div
slot="actions"
class="layout row justify-end"
>
<v-btn
flat
@click="$store.dispatch('popDialogStack')"
>
Done
</v-btn>
</div>
</dialog-base>
</template>
<script>
import LibraryNodes, {
updateLibraryNode,
pushToLibraryNode,
pullFromLibraryNode,
softRemoveLibraryNode,
} from '/imports/api/library/LibraryNodes.js';
import DialogBase from '/imports/ui/dialogStack/DialogBase.vue';
import { getPropertyName } from '/imports/constants/PROPERTIES.js';
import PropertyIcon from '/imports/ui/properties/shared/PropertyIcon.vue';
import propertyFormIndex from '/imports/ui/properties/forms/shared/propertyFormIndex.js';
import { get } from 'lodash';
import { assertDocEditPermission } from '/imports/api/sharing/sharingPermissions.js';
export default {
components: {
...propertyFormIndex,
PropertyIcon,
DialogBase,
},
props: {
_id: String,
},
reactiveProvide: {
name: 'context',
include: ['debounceTime', 'editPermission'],
},
data(){return {
debounceTime: 0,
}},
meteor: {
model(){
return LibraryNodes.findOne(this._id);
},
editPermission(){
try {
assertDocEditPermission(this.model, Meteor.userId());
return true;
} catch (e) {
return false;
}
},
},
methods: {
getPropertyName,
change({path, value, ack}){
updateLibraryNode.call({_id: this._id, path, value}, (error) =>{
ack && ack(error && error.reason || error);
});
},
push({path, value, ack}){
pushToLibraryNode.call({_id: this._id, path, value}, (error) =>{
ack && ack(error && error.reason || error);
});
},
pull({path, ack}){
let itemId = get(this.model, path)._id;
path.pop();
pullFromLibraryNode.call({_id: this._id, path, itemId}, (error) =>{
ack && ack(error && error.reason || error);
});
},
remove(){
softRemoveLibraryNode.call({_id: this._id});
this.$store.dispatch('popDialogStack');
},
}
};
</script>
<style lang="css" scoped>
</style>

View File

@@ -0,0 +1,37 @@
<template lang="html">
<dialog-base>
<v-toolbar-title slot="toolbar">
Select new location
</v-toolbar-title>
<library-and-node
slot="unwrapped-content"
style="height: 100%;"
selection
@selected="val => node = val"
/>
<template slot="actions">
<v-spacer />
<v-btn
flat
color="primary"
@click="$store.dispatch('popDialogStack', node._id)"
>
Move
</v-btn>
</template>
</dialog-base>
</template>
<script>
import DialogBase from '/imports/ui/dialogStack/DialogBase.vue';
import LibraryAndNode from '/imports/ui/library/LibraryAndNode.vue';
export default {
components: {
DialogBase,
LibraryAndNode,
},
data(){return {
node: undefined,
};},
};
</script>

View File

@@ -2,7 +2,7 @@
<character-sheet
show-menu-button
:creature-id="$route.params.id"
:tabs="tabs"
:tabs.sync="activeTab"
/>
</template>
@@ -18,5 +18,15 @@ export default {
required: true,
},
},
computed: {
activeTab: {
get(){
return this.tabs;
},
set(newTab){
this.$emit('update:tabs', newTab);
},
},
},
}
</script>

View File

@@ -1,6 +1,11 @@
<template lang="html">
<div>
<v-card class="ma-4">
<div
class="pa-4 layout column align-center"
style="height: calc(100vh - 96px); display: flex;"
>
<v-card
style="height: 100%; width: 100%; max-width: 1800px;"
>
<library-and-node />
</v-card>
</div>

View File

@@ -1,5 +1,7 @@
<template lang="html">
<v-icon>{{icon}}</v-icon>
<v-icon :color="color">
{{ icon }}
</v-icon>
</template>
<script>
@@ -8,6 +10,7 @@ import { getPropertyIcon } from '/imports/constants/PROPERTIES.js';
export default {
props: {
type: String,
color: String,
},
computed: {
icon(){

View File

@@ -4,6 +4,7 @@
class="mr-2"
:type="model.type"
:class="selected && 'primary--text'"
:color="model.color"
/>
<div
class="text-no-wrap text-truncate"

View File

@@ -4,6 +4,7 @@
class="mr-2"
:type="model.type"
:class="selected && 'primary--text'"
:color="model.color"
/>
<div
class="text-no-wrap text-truncate"

View File

@@ -3,6 +3,7 @@
<property-icon
class="mr-2"
:type="model.type"
:color="model.color"
:class="selected && 'primary--text'"
/>
<div class="text-no-wrap text-truncate">

View File

@@ -3,6 +3,7 @@
<v-icon
class="mr-2"
:class="selected && 'primary--text'"
:color="model.color"
>
{{ effectIcon }}
</v-icon>

View File

@@ -3,6 +3,7 @@
<v-icon
class="mr-2"
:class="selected && 'primary--text'"
:color="model.color"
>
{{ model.equipped ? 'check_box' : 'check_box_outline_blank' }}
</v-icon>

View File

@@ -1,21 +1,21 @@
const theme = {
primary: "#B71C1C",
secondary: "#424242",
accent: "#B71C1C",
error: "#FF6D00",
warning: "#FFB300",
info: "#5C6BC0",
success: "#43A047",
primary: '#B71C1C',
secondary: '#424242',
accent: '#B71C1C',
error: '#FF6D00',
warning: '#FFB300',
info: '#5C6BC0',
success: '#43A047',
};
const darkTheme = {
primary: "#f44336",
secondary: "#424242",
accent: "#f44336",
error: "#FF6D00",
warning: "#FFB300",
info: "#5C6BC0",
success: "#43A047",
primary: '#f44336',
secondary: '#757575',
accent: '#f44336',
error: '#FF6D00',
warning: '#FFB300',
info: '#5C6BC0',
success: '#43A047',
};
export {theme, darkTheme};

View File

@@ -1,7 +1,9 @@
export function kebabToCamelCase(string){
if (!string) return string;
return string.replace(/-([a-z0-9])/g, g => g[1].toUpperCase());
};
}
export function camelToKebabCase(string){
if (!string) return string;
return string.replace(/([a-z][A-Z0-9])/g, g => g[0] + '-' + g[1].toLowerCase());
};
}