mirror of
https://github.com/sojamo/controlp5
synced 2025-01-09 19:55:16 +01:00
188 lines
5.1 KiB
Plaintext
188 lines
5.1 KiB
Plaintext
|
|
||
|
/**
|
||
|
* ControlP5 MenuList
|
||
|
*
|
||
|
* A custom Controller, a scrollable Menu List, using a PGraphics buffer.
|
||
|
* Allows custom designs for List Item.
|
||
|
*
|
||
|
* by Andreas Schlegel, 2013
|
||
|
* www.sojamo.de/libraries/controlp5
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
import controlP5.*;
|
||
|
import static controlP5.ControlP5.*;
|
||
|
import java.util.*;
|
||
|
import java.util.Map.Entry;
|
||
|
|
||
|
|
||
|
ControlP5 cp5;
|
||
|
|
||
|
PFont f1, f2;
|
||
|
void setup() {
|
||
|
size(800, 500, P3D );
|
||
|
f1 = createFont("Helvetica", 20);
|
||
|
f2 = createFont("Helvetica", 12);
|
||
|
|
||
|
cp5 = new ControlP5( this );
|
||
|
|
||
|
|
||
|
/* create a custom MenuList with name menu, notice that function
|
||
|
* menu will be called when a menu item has been clicked.
|
||
|
*/
|
||
|
MenuList m = new MenuList( cp5, "menu", 200, 368 );
|
||
|
|
||
|
m.setPosition(40, 40);
|
||
|
// add some items to our menuList
|
||
|
for (int i=0;i<100;i++) {
|
||
|
m.addItem(makeItem("headline-"+i, "subline", "some copy lorem ipsum ", createImage(50, 50, RGB)));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* a convenience function to build a map that contains our key-value pairs which we will
|
||
|
* then use to render each item of the menuList.
|
||
|
*/
|
||
|
Map<String, Object> makeItem(String theHeadline, String theSubline, String theCopy, PImage theImage) {
|
||
|
Map m = new HashMap<String, Object>();
|
||
|
m.put("headline", theHeadline);
|
||
|
m.put("subline", theSubline);
|
||
|
m.put("copy", theCopy);
|
||
|
m.put("image", theImage);
|
||
|
return m;
|
||
|
}
|
||
|
|
||
|
void menu(int i) {
|
||
|
println("got some menu event from item with index "+i);
|
||
|
}
|
||
|
|
||
|
public void controlEvent(ControlEvent theEvent) {
|
||
|
if(theEvent.isFrom("menu")){
|
||
|
Map m = ((MenuList)theEvent.getController()).getItem(int(theEvent.getValue()));
|
||
|
println("got a menu event from item : "+m);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void draw() {
|
||
|
background( 40 );
|
||
|
}
|
||
|
|
||
|
|
||
|
/* A custom Controller that implements a scrollable menuList. Here the controller
|
||
|
* uses a PGraphics element to render customizable list items. The menuList can be scrolled
|
||
|
* using the scroll-wheel, touchpad, or mouse-drag. Items are triggered by a click. clicking
|
||
|
* the scrollbar to the right makes the list scroll to the item correspoinding to the
|
||
|
* click-location.
|
||
|
*/
|
||
|
class MenuList extends Controller<MenuList> {
|
||
|
|
||
|
float pos, npos;
|
||
|
int itemHeight = 100;
|
||
|
int scrollerLength = 40;
|
||
|
List< Map<String, Object>> items = new ArrayList< Map<String, Object>>();
|
||
|
PGraphics menu;
|
||
|
boolean updateMenu;
|
||
|
|
||
|
MenuList(ControlP5 c, String theName, int theWidth, int theHeight) {
|
||
|
super( c, theName, 0, 0, theWidth, theHeight );
|
||
|
c.register( this );
|
||
|
menu = createGraphics(getWidth(), getHeight() );
|
||
|
|
||
|
setView(new ControllerView<MenuList>() {
|
||
|
|
||
|
public void display(PGraphics pg, MenuList t ) {
|
||
|
if (updateMenu) {
|
||
|
updateMenu();
|
||
|
}
|
||
|
if (inside() ) {
|
||
|
menu.beginDraw();
|
||
|
int len = -(itemHeight * items.size()) + getHeight();
|
||
|
int ty = int(map(pos, len, 0, getHeight() - scrollerLength - 2, 2 ) );
|
||
|
menu.fill(255 );
|
||
|
menu.rect(getWidth()-4, ty, 4, scrollerLength );
|
||
|
menu.endDraw();
|
||
|
}
|
||
|
pg.image(menu, 0, 0);
|
||
|
}
|
||
|
}
|
||
|
);
|
||
|
updateMenu();
|
||
|
}
|
||
|
|
||
|
/* only update the image buffer when necessary - to save some resources */
|
||
|
void updateMenu() {
|
||
|
int len = -(itemHeight * items.size()) + getHeight();
|
||
|
npos = constrain(npos, len, 0);
|
||
|
pos += (npos - pos) * 0.1;
|
||
|
menu.beginDraw();
|
||
|
menu.noStroke();
|
||
|
menu.background(255, 64 );
|
||
|
menu.textFont(cp5.getFont().getFont());
|
||
|
menu.pushMatrix();
|
||
|
menu.translate( 0, pos );
|
||
|
menu.pushMatrix();
|
||
|
|
||
|
int i0 = PApplet.max( 0, int(map(-pos, 0, itemHeight * items.size(), 0, items.size())));
|
||
|
int range = ceil((float(getHeight())/float(itemHeight))+1);
|
||
|
int i1 = PApplet.min( items.size(), i0 + range );
|
||
|
|
||
|
menu.translate(0, i0*itemHeight);
|
||
|
|
||
|
for (int i=i0;i<i1;i++) {
|
||
|
Map m = items.get(i);
|
||
|
menu.fill(255, 100);
|
||
|
menu.rect(0, 0, getWidth(), itemHeight-1 );
|
||
|
menu.fill(255);
|
||
|
menu.textFont(f1);
|
||
|
menu.text(m.get("headline").toString(), 10, 20 );
|
||
|
menu.textFont(f2);
|
||
|
menu.textLeading(12);
|
||
|
menu.text(m.get("subline").toString(), 10, 35 );
|
||
|
menu.text(m.get("copy").toString(), 10, 50, 120, 50 );
|
||
|
menu.image(((PImage)m.get("image")), 140, 10, 50, 50 );
|
||
|
menu.translate( 0, itemHeight );
|
||
|
}
|
||
|
menu.popMatrix();
|
||
|
menu.popMatrix();
|
||
|
menu.endDraw();
|
||
|
updateMenu = abs(npos-pos)>0.01 ? true:false;
|
||
|
}
|
||
|
|
||
|
/* when detecting a click, check if the click happend to the far right, if yes, scroll to that position,
|
||
|
* otherwise do whatever this item of the list is supposed to do.
|
||
|
*/
|
||
|
public void onClick() {
|
||
|
if (getPointer().x()>getWidth()-10) {
|
||
|
npos= -map(getPointer().y(), 0, getHeight(), 0, items.size()*itemHeight);
|
||
|
updateMenu = true;
|
||
|
}
|
||
|
else {
|
||
|
int len = itemHeight * items.size();
|
||
|
int index = int( map( getPointer().y() - pos, 0, len, 0, items.size() ) ) ;
|
||
|
setValue(index);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void onMove() {
|
||
|
}
|
||
|
|
||
|
public void onDrag() {
|
||
|
npos += getPointer().dy() * 2;
|
||
|
updateMenu = true;
|
||
|
}
|
||
|
|
||
|
public void onScroll(int n) {
|
||
|
npos += ( n * 4 );
|
||
|
updateMenu = true;
|
||
|
}
|
||
|
|
||
|
void addItem(Map<String, Object> m) {
|
||
|
items.add(m);
|
||
|
updateMenu = true;
|
||
|
}
|
||
|
|
||
|
Map<String,Object> getItem(int theIndex) {
|
||
|
return items.get(theIndex);
|
||
|
}
|
||
|
}
|
||
|
|