#pragma once #include "ofMain.h" #include "ofxSvg.h" #include "lineTransformer.h" #include "colourPolyline.h" #include "ofxXmlSettings.h" class glyph{ public: glyph(char c,float w,vector lines){ glyph(c,w,lines,ofColor(0,0,0)); } glyph(char c,float w,vector lines,ofColor col){ code=c; width=w; outline=lines; colour=col; } void draw(float x, float y){ ofSetColor(colour); ofPushMatrix(); ofTranslate(x,y); for (auto& v:outline) v.draw(); ofPopMatrix(); } void randomiseColour(){ colour=ofColor::fromHsb(ofRandom(255.0),225,255); } char code; float width; vector outline; ofColor colour; }; class glyphWord{ public: glyphWord(){ amount=1.0f; val=ofRandom(1.0f); val2=ofRandom(1.0f); width=0; } vector glyphs; float amount; float val; //a random float used for word parameters float val2; float width; }; #define STYLE_WORDS 0 #define STYLE_OVERLAPPING 1 #define STYLE_SENTENCE 2 class glyphbanner{ ofXml SVGFont; vector words; vector outlines; ofVec2f centre; float lastUpdateTime; float playhead; float enspace; float last_beat; float last_beat_duration; vector palette; struct { vector balanced={ ofColor::fromHex(0xE27D60), ofColor::fromHex(0x085DCB), ofColor::fromHex(0xE8A87C), ofColor::fromHex(0xC38D9E), ofColor::fromHex(0x41B3A3) }; vector uneasy={ ofColor::fromHex(0x2154B9), ofColor::fromHex(0xC45A62), ofColor::fromHex(0x95A28A), ofColor::fromHex(0x98546D), ofColor::fromHex(0xE9DADA), ofColor::fromHex(0x9FF3E9), ofColor::fromHex(0xD07B37), ofColor::fromHex(0x741710), ofColor::fromHex(0x102ADC), ofColor::fromHex(0x9FA1AC) }; vector stark={ ofColor::fromHex(0xFFFFFF), ofColor::fromHex(0x000000), ofColor::fromHex(0x000000), ofColor::fromHex(0x000000) }; }palettes; vector split(string s) { size_t pos_start = 0, pos_end; string token; vector res; while ((pos_end = s.find (" ", pos_start)) != string::npos) { token = s.substr (pos_start, pos_end - pos_start); pos_start = pos_end + 1; res.push_back (token); } res.push_back (s.substr (pos_start)); return res; } float segment; //the fraction that the word has displayed public: glyphbanner(){ palette=palettes.stark; }; void init(string message){ createWords(message); lastUpdateTime=ofGetElapsedTimef(); playhead=0.0f; } int length(){ int l=0; for (auto& w:words) { l+=w.glyphs.size(); } return l+max(0,(int)words.size()-1); } float width(){ float _w=0.0f; for (auto& w:words) { for (auto& g:w.glyphs) _w+=g.width; } return _w+max((float)(words.size()-1)*enspace,0.0f); } int glyphCount(){ int c=0; for (auto& w:words){ c+=w.glyphs.size(); } return c; } string text(){ string s; for (auto& w:words) { for (auto& g:w.glyphs) s+=ofToString(g.code); } return s; } void loadFont(filesystem::path path){ if( SVGFont.load(path) ){ vector w=words; clear(); createWords(w); enspace=getGlyph(' ').width; ofLog()<<"loaded "< p){ palette=p; } void loadPalette(string path){ ofxXmlSettings xml; if (xml.load(path)){ ofLog()< palette; for (int i=0;i m=split(message); for (auto& word: m){ glyphWord w; for (auto& c: word){ glyph g=getGlyph(c, usePalette? palette[ofRandom(palette.size())]: //112->231 hue sat 0->255 brightness 255 ofColor::fromHsb(ofRandom(119)+112,ofRandom(255),255) ); w.width+=g.width; w.glyphs.push_back(g); } words.push_back(w); } ofLog()<<"glyphbanner created "< _words){ clear(); for (auto& _w:_words) { glyphWord w; for (auto& g: _w.glyphs){ w.glyphs.push_back(getGlyph(g.code,g.colour)); } words.push_back(w); } } glyph getGlyph(char c,ofColor col=ofColor(255,255,255)){ vector shapes; ofPolyline shape; string elementPath = "/svg/defs/font/glyph[@unicode='"+ofToString(c)+"']"; if(SVGFont.findFirst(elementPath) == 0 ){ elementPath = "/svg/defs/font/glyph[@unicode='?']"; } ofXml xmlElement = SVGFont.findFirst(elementPath); float charWidth = ofToFloat(xmlElement.getAttribute("horiz-adv-x").getValue()); vector splitGlyphPath = ofSplitString(xmlElement.getAttribute("d").getValue(), " ");//glyph path data in SVG looks like this: "M 139 -9.45 L 230 18.9 L 299 22.1 L 227 25.2" for(int i=0; iwords[theword].glyphs.size()){ theletter-=words[theword].glyphs.size(); theword++; } float playfraction=playhead-int(playhead); //segment=(((float)theletter+words[theword].glyphs.size()+playhead-int(playhead))/words[theword].glyphs.size()); segment=(((float)theletter+playfraction-1)/words[theword].glyphs.size()); } //calculate params for word/letter anim for (int i=0;i& getOutlines(float s=1.0f,int style=STYLE_SENTENCE,float x=0,float y=0,bool anim=false,bool reverse=false, float vert_spread=0.0f, bool use_beat=false, float beat_duration=0.0f){ outlines.clear(); //if (beat_duration>segment){ //I don't think this is what segment is // return outlines; //} int drawglyphs=0; int drawlines=0; float a=anim?reverse?1.0f-segment:segment:1.0; switch (style){ case STYLE_WORDS:{ for (auto& w:words){ float p=(w.val*(2.0-w.width)); //was screen based now 2.0 float v=y+(vert_spread*w.val*(2.0)); //was screen based now 2.0 for (auto& g:w.glyphs){ if (w.amount>0.0f&&g.colour.getBrightness()>0){ drawglyphs++; for (auto& o:g.outline){ drawlines++; auto q=o; q.scale(s*a,-s*a); q.translate(glm::vec3((p*s*a)+x,v,0)); outlines.push_back(colourPolyline(q,g.colour*w.amount)); } } p+=g.width; } } break; } case STYLE_OVERLAPPING:{ for (auto& w:words){ float p=(w.val*(2.0-w.width)); //was screen based now 2.0 float v=y+(vert_spread*w.val*(2.0)); //was screen based now 2.0 for (auto& g:w.glyphs){ if (w.amount>0.0f&&g.colour.getBrightness()>0){ drawglyphs++; for (auto& o:g.outline){ drawlines++; auto q=o; q.scale(s*a,-s*a); q.translate(glm::vec3((p*s*a)+x,v,0)); outlines.push_back(colourPolyline(q,g.colour*w.amount)); } } p+=g.width*(anim?a:1.0); } } break; } case STYLE_SENTENCE:{ float p=((-width())/2); for (auto& w:words){ for (auto& g:w.glyphs){ if (w.amount>0.0f&&g.colour.getBrightness()>0){ for (auto& o:g.outline){ auto q=o; q.scale(s,-s); q.translate(glm::vec3((p*s)+x,y,0)); outlines.push_back(colourPolyline(q,g.colour*w.amount)); } } p+=s*g.width; } p+=s*enspace; } break; } default:{ break; } } //ofLog()<<"outlines for "<