Skip to content

pixelsandcandy/ofxVui

Repository files navigation

✨ Much Easy! View + GUI System for openFrameworks ✨

BRANCHES
  • master » Latest version (constantly adding features, use at your own risk 😈)
  • v1-stable » Last commit 12/21/2018 (use this or commit 0f429e641b8e5554c7e773e14fd50910fd28d451)

TELEPORT


  • openFrameworks 0.9.8
    • macOS High Sierra (10.13) » Xcode 9 (✔️ Debug/Release)
    • Windows 10 » Visual Studio 2017 (✔️)
  • openFrameworks 0.10.1
    • macOS High Sierra (10.13) » Xcode 9 (❌)
    • macOS High Sierra (10.13) » Xcode 10 (⚠️ Release Build Only)

  • Xcode » openFrameworks-Info.plist » right-click » Add Row » High Resolution Capable » YES
    • in main.cpp » first line in main() » VUI::SetDpi(2.0)

So everything works! 🎉 Although, the code under the hood is a little bit of spaghetti at the moment, uncommented, and there's a lot of legacy/debug stuff 😅. I was originally working on two different addons but decided to merge them together. Cleaning up the code is on my TODO list but first wanted to make this addon usable, tested, and make enough project/code examples. I have to get back to other projects though so yea... please feel free to enjoy this as-is 😋 and I'll continue to make improvements/add features whenever I get the chance 😊

Also, ofxVui uses Pascal Case (UpperCamelCase) for methods because I've grown fond of it in C#/Unity - it just seems more legible IMO 🍕


ofxVui Video


View Manager

  • Easily add/set views from anywhere
  • Views have access to same methods as ofApp
  • Mouse/Touch Events still work when scaling/rotating view
Resolution
Scale
Rotation
  ∟ VUI_ROTATE_90_CCW

GUI System

  • Mouse/Touch events respect Element render order
    • Event sends local and global Mouse/Touch positions
  • Easily animate Elements using one call
    • Tween has START, STEP, COMPLETE events
  • Easily style/texture Elements per state » UP, OVER, DOWN
  • StyleSheet manages fonts/images so there's no redundant loading
Elements
  ∟ Element
  ∟ Element->MakeToggle()
  ∟ ToggleGroup
  ∟ Text
  ∟ TextField
  ∟ Slider
  ∟ Container

  (x/y position » px, %, calc)

Animation Tween
  ∟ x
  ∟ y
  ∟ width
  ∟ height
  ∟ scale
  ∟ opacity
  ∟ rotation
  ∟ ease/easing

  (x/y/width/height/scale/opacity/rotation » px, +=, -=, *=)

Animation Events
  ∟ ANIMATE_START
  ∟ ANIMATE_STEP
  ∟ ANIMATE_COMPLETE

Mouse Events
  ∟ MOUSE_OVER
  ∟ MOUSE_OUT
  ∟ MOUSE_PRESSED
  ∟ MOUSE_MOVED
  ∟ MOUSE_DRAGGED
  ∟ MOUSE_RELEASED
  ∟ MOUSE_CLICK
  ∟ MOUSE_DOUBLE_CLICK

Touch Events (only single touch usage tested so far)
  ∟ TOUCH_DOWN
  ∟ TOUCH_UP
  ∟ TOUCH_TAP
  ∟ TOUCH_DOUBLE_TAP

Other Events
  ∟ STATE_CHANGE
  ∟ TEXT_CHANGE <Text/TextField>
  ∟ VALUE_CHANGE <Element->MakeToggle()>
  ∟ TOGGLE_CHANGE <ToggleGroup>
  ∟ SUBMIT <TextField>
  ∟ FOCUS <TextField>
  ∟ UNFOCUS <TextField>

GUI States
  ∟ STATE_UP
  ∟ STATE_OVER
  ∟ STATE_DOWN

GUI StyleSheet: Styles
  ∟ width » px/%/calc
  ∟ height » px/%/calc
  ∟ scale
  ∟ opacity
  ∟ rotation
  ∟ backgroundImage
  ∟ backgroundColor
  ∟ anchorPoint
  ∟ offset
  ∟ color <Text/TextField>
  ∟ padding <Text/TextField>
  ∟ textAlign <Text/TextField>
  ∟ font <Text/TextField>

GUI Alignments for anchorPoint/textAlign
  ∟ LEFT_TOP
  ∟ LEFT_CENTER
  ∟ LEFT_BOTTOM
  ∟ CENTER_TOP
  ∟ CENTER_CENTER
  ∟ CENTER_BOTTOM
  ∟ RIGHT_TOP
  ∟ RIGHT_CENTER
  ∟ RIGHT_BOTTOM

There are a bunch of example projects/code in the repo but here are some basics.

For more info checkout the example-Animation project/code

  • Animate( Float seconds, String params );
#include "ofxVui.h"
using namespace VUI;

// basic
elementPtr->Animate(1.25, "{x: +=160, width: 150, height: 50, scale: *=.5, rotation: -=90, opacity: .75, ease: Elastic.easeOut }");


// add listener - method A
elementPtr->Animate(1.25, "{x: 160, width: 150, height: 50, scale: 1.75, rotation: -135, opacity: .75, ease: Elastic.easeOut }", this, &ListenerClass::vuiEventHandler);


// add listener - method B
Tween* tweenPtr = elementPtr->Animate(1.25, "{x: 160, width: 150, height: 50, scale: 1.75, rotation: -135, opacity: .75, ease: Elastic.easeOut }");
ofAddListener( tweenPtr->onStep, this, &ListenerClass::vuiEventHandler );
ofAddListener( tweenPtr->onComplete, this, &ListenerClass::vuiEventHandler );


// can also set/reset animation properties:
elementPtr->Set("{x: 0, width: 100, height: 100, scale: 1, rotation: 0, opacity: 1}");


// eventHandler
void ListenerClass::vuiEventHandler(vuiEventArgs& evt){
  if ( evt.element == elementPtr ){
    if ( evt.eventType == VUI_EVENT_ANIMATE_COMPLETE ) /* do stuff */;
    else if ( evt.eventType == VUI_EVENT_ANIMATE_STEP ){
      ofLog() << evt.tween->GetProgress();
    }
  }
}


/* -------------------------------------------- easing functions

  Back.easeIn
  Back.easeOut
  Back.easeInOut
  Bounce.easeIn
  Bounce.easeOut
  Bounce.easeInOut
  Circ.easeIn
  Circ.easeOut
  Circ.easeInOut
  Cubic.easeIn
  Cubic.easeOut
  Cubic.easeInOut
  Elastic.easeIn
  Elastic.easeOut
  Elastic.easeInOut
  Exp.easeIn
  Exp.easeOut
  Exp.easeInOut
  Linear.easeIn
  Linear.easeOut
  Linear.easeInOut
  Linear.easeNone
  Quad.easeIn
  Quad.easeOut
  Quad.easeInOut

*/

For more info checkout the example-Events project/code

#include "ofxVui.h"
using namespace VUI;

/*
  Some Mouse/Touch events are interchangeable so if you
  use VUI::EnableTouch() you don't have to change your code:
    - onMousePressed / onTouchDown
    - onMouseReleased / onTouchUp
    - onMouseClick / onTouchTap
    - onMouseDoubleClick / onTouchDoubleTap
*/

// add listener
ofAddListener( elementPtr->onMouseClick, this, &ListenerClass::vuiEventHandler );


// eventHandler
void ListenerClass::vuiEventHandler(vuiEventArgs& evt){
  if ( evt.element == elementPtr ){
    if ( evt.eventType == VUI_EVENT_MOUSE_CLICK ) {
      ofLog() << evt.localMousePos;
    };
  }
}


/* -------------------------------------------- vuiEventArgs

  Element* element;
  int eventType;
  int renderState;
  int virtualState;

  int value = -1;
  string text = "";

  ofVec2f localMousePos;
  ofVec2f localDragDelta;
  ofVec2f localDragStart;

  ofVec2f globalMousePos;

  Tween* tween;

*/

/* -------------------------------------------- events

  VUI_EVENT_MOUSE_OVER
  VUI_EVENT_MOUSE_OUT
  VUI_EVENT_MOUSE_PRESSED
  VUI_EVENT_MOUSE_MOVED
  VUI_EVENT_MOUSE_DRAGGED
  VUI_EVENT_MOUSE_RELEASED
  VUI_EVENT_MOUSE_CLICK
  VUI_EVENT_MOUSE_DOUBLE_CLICK

  VUI_EVENT_TOUCH_DOWN
  VUI_EVENT_TOUCH_UP
  VUI_EVENT_TOUCH_TAP
  VUI_EVENT_TOUCH_DOUBLE_TAP

  VUI_EVENT_STATE_CHANGE
  VUI_EVENT_TEXT_CHANGE
  VUI_EVENT_VALUE_CHANGE
  VUI_EVENT_TOGGLE_CHANGE
  VUI_EVENT_SLIDER_VALUE_CHANGE
  VUI_EVENT_SUBMIT

  VUI_EVENT_FOCUS
  VUI_EVENT_UNFOCUS

  VUI_EVENT_ANIMATE_COMPLETE
  VUI_EVENT_ANIMATE_STEP
  VUI_EVENT_ANIMATE_START

*/

For more info checkout the example-StylingElements project/code

#include "ofxVui.h"
using namespace VUI;

StyleSheet* ss;
Element* buttonA;
Element* buttonB;
Text* label;


// --------------------------------------------

void setup(){
  string styles = R"(
    [Images>
      btn-up: path/to/image.jpg;
      btn-over: path/to/image-over.jpg;
      btn-down: path/to/image-down.jpg;
    ]

    [.button>
      width: calc(40%-20);
      height: 10%;
      bgImage: btn-up, FILL;

      &:over{
        bgImage: btn-over, FILL;
      }

      &:down{
        bgImage: btn-down, FILL;
      }
    ]

    [#buttonB>
      scale: 1.25;
      opacity: .9;
    ]

    [.text>
      width: 160;
      height: 32;
      font: path/to/fontfile.ttf,16;
      color: #dddddd;
      padding: 10;
      anchorPoint: center-center;
      textAlign: center-center;
      bg: #ffffff;
    ]
  )";

  //

  ss = new StyleSheet( styles );

  /*

    name stylesheet if you want to load/use it elsewhere

    ss = new StyleSheet( styles, "dope-stylesheet" );
    StyleSheet *elsewhere = VUI::GetStyleSheet( "dope-stylesheet" );

  */


  label = new Text( "50%", "50%", ss, ".text" );

  // true = automatically set width/height to text
  label->SetText( "Stuff", true );


  // Element(x, y, StyleSheet*, primarySelector, secondarySelector );
  buttonA = new Element( 20, "calc(50%+30)", ss, ".button" );
  buttonB = new Element( "10%", 300, ss, ".button", "#buttonB" );

}


// --------------------------------------------

void draw(){
  label->Render();
  buttonA->Render();
  buttonB->Render();
}

For more info checkout the example-ViewManager project/code

#include "ofxVui.h"
using namespace VUI;

// ---------------------------------------- main.cpp

VUI::SetResolution(1080,1920,.6);
// VUI::RotateView( VUI_ROTATE_90_CCW );

ofSetupOpenGL(VUI::GetWindowWidth(),VUI::GetWindowHeight(),OF_WINDOW);


// ---------------------------------------- create View

#include "ofxVui.h"
using namespace VUI;

class MyView : public View {
    // a few extra stuff
    /*
       if overriding BeforeExitView(), you must call ExitView()
       when you're done with what you need to do
     */
    virtual void BeforeExitView() {
      ExitView();
    };

    virtual void OnEnterView() {};

    // basic stuff
    int mouseX;
    int mouseY;

    void setup(){};
    void update(){};
    void draw(){};

    void keyPressed(int key){};
    void keyReleased(int key){};
    void mouseMoved(int x, int y ){};
    void mouseDragged(int x, int y, int button){};
    void mousePressed(int x, int y, int button){};
    void mouseReleased(int x, int y, int button){};
    void mouseEntered(int x, int y){};
    void mouseExited(int x, int y){};
    void windowResized(int w, int h){};
    void dragEvent(ofDragInfo dragInfo){};
    void gotMessage(ofMessage msg){};

}


// ---------------------------------------- ofApp.cpp

#include "MyView.h"
#include "MyOtherView.h"

void setup(){
  //Windows only, please see Dependencies section above for more info
  //VUI::EnableTouch();

  VUI::AddView( "view-name", new MyView() );
  VUI::AddView( "my-other-view", new MyOtherView() );


  // can call from anywhere
  VUI::SetView( "view-name" );
}

void draw(){
  /*
      VUI::Render() is VUI::RenderBegin() + VUI::RenderEnd()
  */
  VUI::Render();


  /*

      If you still want to scale/rotate your view without using the
      View Manager (via AddView()/SetView() you can like this:

          VUI::RenderBegin();
            // draw your stuff here
          VUI::RenderEnd();


      But you MUST set the 4th arg in VUI::SetResolution() to false (main.cpp):
      (ofxVui Element's Mouse/Touch events will adapt to scaled/rotated coordinates)

          VUI::SetResolution(w, h, scale, enableViewManager );
          eg: VUI::SetResolution(1920, 1080, .6, false);

   */
}

  ⭐️ Touch
      + Test multiple touch events / scenarios
      + Add - VUI_EVENT_TOUCH_DRAG

  ⭐️ UI Elements
      ✔️ Add - Container - scrollable
      ✔️ Add - Slider
      + Add - Slider2D

  ⭐️ StyleSheet
      ✔️ width/height - percentage based values
      ✔️ width/height - calc() based values

  ⭐️ Animate
      ✔️ += value (x/y/width/height/scale/opacity/rotation)
      ✔️ -= value (x/y/width/height/scale/opacity/rotation)
      ✔️ *= value (x/y/width/height/scale/opacity/rotation)

  ⭐️ VUI::SetRotation()
      + Add option - VUI_ROTATE_90_CW
      + Add option - VUI_ROTATE_180

  ⭐️ Cleanup / add comments to ofxVui code
  ⭐️ Add more emojis to this README 👓

If you think of a feature that would be super useful in this addon or have a request to prioritize a feature, please feel free to email me » [email protected] -or- submit a pull request :)

Releases

No releases published

Packages

No packages published

Languages