A Pie- and Marking-Menu Framework in TypeScript.
Add tasty.js
to your HTML File.
You'll need to include paper.js and lodash separately.
<script src="https://unpkg.com/lodash"></script>
<script src="https://unpkg.com/paper"></script>
The framework is accessible through the global tasty
object.
Example Menu:
window.onload = () => {
const menu = new tasty.menu('body' /* The element to place the menu into */, {
// Configuration object
// These are the defaults
main: {
minDistance: 150,
minTraceDistance: 175,
animationDuration: 250,
enableMaxClickRadius: true,
},
geometry: {
size: 50,
color: '#575859',
selectionColor: '#577a85',
stroke: {
enabled: true,
color: 'rgba(62, 62, 64, 1.0)',
width: 2,
},
text: {
color: '#fff',
},
icon: {
color: '#fff',
}
},
connector: {
enabled: true,
color: '#393a3c',
width: 10,
},
arc: {
enabled: true,
color: [
['#575859', 0.15],
['rgba(87, 122, 133, 0.5)', 0.5],
['rgba(74, 159, 158, 0.15)', 0.8],
'rgba(87, 88, 89, 0)'
],
radial: true,
stroke: {
enabled: true,
width: 1,
}
},
scales: {
parent: 0.5,
child: 0.5,
dot: 0.15,
icon: {
base: 0.0625,
solo: 1.6,
child: 0.7,
}
},
radii: {
maxClickRadius: 450,
}
});
const structure = {
// There can only be one topmost menu item!
id: 'Root',
text: 'Hauptmenü',
direction: 0,
icon: 'bars',
// Add your Items here
children: [
{
id: 'file',
text: 'Datei',
icon: 'file',
direction: 0,
},
{
id: 'tools',
text: 'Werkzeuge',
icon: 'tools',
direction: 90,
children: [],
},
{
id: 'project',
text: 'Projekt',
icon: 'project-diagram',
direction: 180,
},
{
id: 'documents',
text: 'Dokumente',
icon: 'file-signature',
direction: 135,
}
]
};
menu.init();
const parser = new tasty.parser();
menu.setStructure(parser.parse(structure));
menu.selection$.subscribe(s => {
if (s.type === 'itemSelection') {
console.log(s);
}
});
}
The menu structure is defined in JSON.
id
: Unique menu identifiertext
: The item's label textdirection
: Item position in degreesicon
: Font-Aweseome icon namechildren
: Optional array with item definitions
{
// There can only be one topmost menu item!
id: 'Root',
text: 'Hauptmenü',
direction: 0,
icon: 'bars',
// Add your Items here
children: [
{
id: 'file',
text: 'Datei',
icon: 'file',
direction: 0,
},
{
id: 'tools',
text: 'Werkzeuge',
icon: 'tools',
direction: 90,
children: [],
},
{
id: 'project',
text: 'Projekt',
icon: 'project-diagram',
direction: 180,
},
{
id: 'documents',
text: 'Dokumente',
icon: 'file-signature',
direction: 270,
}
]
}
An observable communicates the current menu status.
menu.selection$.subscribe(event => {
console.log(event);
});
There are six different event types:
- itemSelection: An item with no children was selected
- navigationSubmenu: A navigation into a submenu has occurred
- navigationBack: A navigation back to the parent has occurred
- navigationBackHover: The input device hovers over the back sector
- itemHover: The input device hovers over an item with children
- itemSelectionHover: The input device hovers over an item with no children
- sliderValueChanging: Ribbonslider value is changing
- sliderValueFinal: Ribbonslider value is finalized
Each event object contains three readonly members:
type
: one of the six described typessource
: An object containing the currently activeitemId
and the item'sangle
target
: An object containing the target'sitemId
andangle
Definition:
{
id: 'slider',
text: 'Ribbonslider',
icon: 'sliders-h',
direction: 0,
// Indicates the ribbonslider class
type: 'ribbonslider',
data: {
min: -100,
max: 100,
initial: 0,
stepSize: 10,
stepDist: 200
}
}
The slider configurations is contained in its data
field.
min
: Slider min value | Cannot be greater than or equal to maxmax
: Slider max value | Cannot be lower than mininitial
: Initial slider value | Cannot be outside the range of min/maxstepSize
: Slider step size between valuesstepDist
: Distance in px between two steps
Definition:
{
id: 'checkbox',
text: 'Checkbox',
icon: 'check',
direction: 0,
type: 'checkbox',
data: {
selected: false
}
}
The itemSelection
event will contain a data
field with selected set to true/false.
Definition:
{
id: 'radio-group',
text: 'Radio-Group',
icon: 'list',
direction: 0,
type: 'radio-group',
// Every added child will be a checkbox
children: [],
}
Radio-Groups will set all added children to be of type checkbox
. Only on item can be active at a given time in a radio-group.