xsteadfastxhttp://xsteadfastx.org/2017-04-20T11:12:00+02:00Nürnberg: Ein Wiedersehen2017-04-20T11:12:00+02:002017-04-20T11:12:00+02:00marvintag:xsteadfastx.org,2017-04-20:/2017/04/20/nurnberg-ein-wiedersehen/<p>Unser erster Nachmittag in Nürnberg seit fast einem Jahr. Als letztes packten wir unsere materialisierten Erinnerungen zusammen, putzten unsere Wohnung und übergaben unsere Schlüssel. "Nürnberg" bedeutete das erste mal "von Zuhause wegziehen". Das erste mal lieben. Wir ließen es zurück um zurück zu kehren. Dorthin wo ich aufgewachsen und geboren …</p><p>Unser erster Nachmittag in Nürnberg seit fast einem Jahr. Als letztes packten wir unsere materialisierten Erinnerungen zusammen, putzten unsere Wohnung und übergaben unsere Schlüssel. "Nürnberg" bedeutete das erste mal "von Zuhause wegziehen". Das erste mal lieben. Wir ließen es zurück um zurück zu kehren. Dorthin wo ich aufgewachsen und geboren bin. Dorthin wo ich nie wieder leben wollte. Mein Herz schien zu verstummen, als wir die Autobahn befuhren. Den Frankenschnellweg entlang und die letzten Meter im Stadtgebiet. Wir zogen in den Norden, wir zogen in ein Haus, unser Haus, und das Leben fuhr fort. Ich versuche jeden Tag mein Fernweh, und zugleich Heimweh, zu vergessen, es zu verdrängen, weit weg zuschieben, es zu entlieben. Hoffnungslos. Ich stelle mir Nürnberg vor, wie es eingefroren, unter eine Glasglocke da steht. Rund angeordnet, in der Mitte der Berg mit der Burg, ein Miniaturwunderland. Ohne mich, nicht mehr lebensfähig. Und jetzt weiß ich, Nürnberg braucht mich nicht. Es hat mich nie gebraucht. Die einzige Glasglocke ist die Glasglocke über meinem Kopf. Manchmal raubt sie mir die Luft zum atmen. Ich habe diese Stadt fast ein Jahr gemieden und hatte den utopischen, selbstheilenden, Plan nie wieder ihren Boden zu betreten. Nicht weil sie mich verletzt hat. Ich habe mich durch ihren Verlust selber verletzt. Nun war es soweit und ich wagte eine neue Annährung. Offenes Visier und mit der Gewissheit alles zu riskieren. Ein Hotel gebucht und auf die Autobahn. Als wir die Brücke der Frankenwald Raststätte unterfuhren, tanzte mein Herz. Nur noch 90 Minuten. Wir fuhren Langwasser ab. Die Ausfahrt die 5 Jahre lang "Zuhause" bedeutete. Christine sagte: "Wir können um eine Stadt genauso trauern, wie um einen Menschen". Vorbei an dem Burger King mit dem abgeschlagenen Reichsadler. Dem chinesischen Restaurant, welches uns auch nach 23 Uhr, Montags, mit Tofu versorgte. Vorbei am Bäcker, in dem wir unsere Sonntagmorgende verbrachten. Die Supermärkte die ich jeden Tag besuchte, um mich mit frischer Cola Light auszustatten. Und endlich, die Straße in der wir wohnten. Wir hielten Ausschau nach all den kleinen Veränderungen. Jede Baustelle und jedes Geschäft. Und es war wie jede andere Fahrt zuvor, vom Dutzendteich in die Innenstadt. Das Hotel liegt am Bahnhof. Dort wo wir uns das erste mal gesehen haben. Dort hat sich damals alles für mich verändert. Wir checkten ein und gingen in die Stadt. Ich hielt mich an Christines Hand fest. Ich drohte an ihr abzurutschen, wegzugleiten. Durch den Bahnhof entlang der Vitrine, ein Ausstellungsort der Kunstakademie in Nürnberg. Ich erkannte die Gesichter der Obdachlosen auf den Treppen zum K4. Die Nacht war lauwarm, die Menschen um herum liefen mit ihren Smartphones vor den Gesichtern herum. Pokemon Go.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/16590125753/"><img src="https://farm9.staticflickr.com/8782/16590125753_4ca7a2d80f_b.jpg" alt="Pegnitz"></a></p> <p>Wir hatten Hunger und sahen das Istanbul. Das beste türkische Restaurant in dem ich je gegessen habe. Nicht das ich in vielen bis jetzt war. Vorstellen konnte ich es mir doch. So gut war das Essen hier. Ich brauche mir nicht einzubilden das ich in dieser Stadt immer noch wohne. Seit ich den Bahnhof verlassen habe und meine tägliche Route in Altstadt betrat, war ich immer noch da. Würde ich nach Hause gehen wollen, würden ich die Straßenbahn Richtung Dokumentationszentrum nehmen, eine halbe Runde um den Dutzendteich laufen, die Menschen im Biergarten sehen, die Liebespaare am Ufer, die thematisch dem Reichsparteigelände angemessenen Hochzeitsfotoshootings, die Alkoholiker am Imbißstand. Dann würde ich die Haustür aufsperren, zwei Stockwerke überwinden und unseren Kater davon abhalten die Treppe Richtung Haustür hinunter zu steigen. Nun setzten uns nach draussen um die Wärme des aufgeheizten Innenstadt Asphalts zu genießen. Hummus und einen Bauernsalat, so wie immer. Mir fielen die Menschen auf die ihre Blicke nicht von den Handys lassen konnten. Türkische Nachrichten und diesmal anscheinend kein Pokemon Go. Ein Putschversuch des türkischen Militärs. Eine Nacht vorher der Terroranschlag auf eine Menschenmenge, die nur das Feuerwerk zum Nationalfeiertag bestaunen wollten. Die BILD würde titeln: "WANN ENDET ENDLICH DIESER FLUCH: Christine und Marvin verreisen". Ende letzten Jahres, als wir gerade in einem Hotel in München uns eingecheckt hatten, wir gerade noch ein Sandwich auf der anderen Straßenseite, im Ost-Bahnhof, gegessen hatten und Terroristen zeitgleich Menschen in Paris erschossen. Unser geliebtes Paris. Wir lagen die ganze Nacht wach und lauschten Deutschlandfunk, als wir erschöpft von der Fahrt, unsere Hände haltend, einschliefen. Oder im Frühling. Wir waren gerade in Paris angekommen. Beim Anziehen, um als peinlicher Tourist die Straßen der Stadt zu betreten, um von einem Leben in dieser Stadt zu träumen, hörten wir von dem Attentat in Brüssel. Nun also wurde unsere Rückkehr mit einen Militärputsch kommentiert. Wir hatten uns rausgesetzt. Ich wollte die Stadt mit ihren Menschen sehen. Ich fühlte mich unsichtbar. So wie die Menschen an uns vorbei zogen, es wirkte alles unantastbar, passiv auf mich. Wie ein Theaterstück, das nur für uns aufgeführt wurde, bei dem wir nur die Zuschauer waren und vor einem Jahr aufgehört haben Darsteller zu sein. Wir aßen unseren Hummus, den Bauernsalat mit warmen und wunderbar weichen Fladenbrot. Als wir bezahlten fragte Christine die Bedienung ob er die aktuelle Lage in der Türkei verfolgt. Er antwortete trocken "Erdogan muss sterben". Wir nahmen uns an die Hand und gingen durch die erleuchtete, freitägliche Stadt. Hinunter zur Pegnitz. Die Temperatur war wunderbar mild und die Teenager waren auf dem Weg in die Disco, oder halt dort wo junge Menschen halt so hingehen. Selbst als ich jung war, wußte ich nie wo das war. Meine Augen fielen fast zu, mein Gehirn schwer, herrenlos eingetaucht in einen See aus einer Melange aus alten Eindrücken, vergangenen Situationen, fest eingebrandt, und der Erkenntnis, das Nürnberg gleich geblieben war. Im Hotel schliefen wir zu den Live-Berichten aus Istanbul ein. Ich traute mich nicht den Fernseher auszuschalten.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/15734440630/"><img src="https://farm8.staticflickr.com/7549/15734440630_a0f3a79252_b.jpg" alt="Nürnberg"></a></p> <p>Der Plan war all unsere Lieblingsorte zu sehen. Natürlich schränkte sich die Liste auf einen geografischen Bereich ein. So hatten wir die Chance einiges sehen zu können. Was akribisch klingt war eine Vorsichtsmaßnahme um sich treiben lassen zu können. Die kleinen Orte und Plätze zu bereisen, die uns so am Herzen liegen. Meine Skepsis war riesig. Der Bäcker, an dem wir sonst öfters frühstückten gab es nicht mehr. Ein kleines Detail auf unserer Reise. Ein Brötchen zu essen mit dem Blick auf den Spalt der zu dem neuen Museum leitet. Wir suchten weiter und gingen leicht widerwillig in einen anderen Bäcker, in dem wir zuvor nicht sehr oft gegessen haben. Eine Prioritätenliste muss es geben, sonst spült uns unsere Nostalgie die Stadt hinunter. Und es würde nächstes mal noch schwieriger sein zu überleben. Ich schlug vor in den Buchladen des neuen Museums zu gehen. Hier verbrachten wir einst immer Stunden. Oft kauften wir nichts, da Ausstellungskataloge der finanzielle Ruin bedeuten. So bräuchte man eigentlich alle, fängt man damit an. Doch all die Bilder die wir sahen, all die wir in unseren Herzen speichern konnten, konnte uns niemand mehr eintreißen. Sie verblassen vielleicht, ordnen sich neu an, doch ihre Schönheit am Tage als wir sie sahen, bleibt für immer. Christine suchte eine Ferienlektüre, ein Buch von Focault. Ich interessierte mich für die neuen Fotografiebände und die Sonderangebote. Erstmal blieb von der Zeit in dem Laden nichts in unseren Taschen zurück. Nürnberg heizte sich weiter auf, so weiter wir zogen.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/12304135996/"><img src="https://farm4.staticflickr.com/3771/12304135996_d674cc11c1_b.jpg" alt="Dutzendteich IX"></a></p> <p>Christine fand ein Geburtstagsgeschenk und wir schwammen weiter und weiter. Dieser Comicladen, bei fast jeden Besuch in der Innenstadt verbrachte ich Zeit dort. Hier fühlte ich mich sehr wohl, auch wenn ich von all dem kaum etwas verstand. Ich las und durchblätterte immer das was mir gefiel. Eins meiner liebsten Bücher ist <a href="https://de.wikipedia.org/wiki/Blankets">Blankets</a> von Craig Thompson. Hätte ich Christine nie kennengelernt, würde jetzt ein Motiv aus diesem Buch meinen Körper zieren. Irgendeine monumentale Zeichnung einer Doppelseite dieses Buchses. Ich glaube ich nicht ich den Schmerz ertragen hätte, doch war ich fest davon überzeugt, dies wäre ein gutes, mit Persönlichkeit aufgeladenes erstes Tattoo. Es sollte mich daran erinnern wie hoffnungslos die Liebe ist und Vergangenheit nicht heilen kann. Diese Erlebnisse würden nie unseren Körper, durch all seine Öffnungen, verlassen können. Das Erscheinen neuer Bücher von ihm markieren immer wieder bestimmte Stellen in meiner Biografie, im Kampf gegen das erwachsen werden. Sein neustes Buch wanderte in unsere heilige Artefaktensammlung.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/9033503951/"><img src="https://farm6.staticflickr.com/5531/9033503951_8eea91670c_b.jpg" alt="Die Burg"></a></p> <p>In Wolfsburg gibt es keine Plattenläden. Die Kundschaft die sich Schallplatten als Dekorationsmittel in den Wohnzimmerschrank stellt, wird gut von Media Markt und Müller versorgt. An einem Winterabend, im ersten Winter in Nürnberg, bog ich einmal falsch ab und stand in einer Straße in der sich drei Plattenläden befanden. Meine Augen und Mund öffneten sich, die Pupillen weiteten sich und das Adrenalin eines Abhängigen schoss in die Bereiche des Gehirns in dem das unstillbare Verlangen verortet war. Ein Laden hieß <a href="http://musicandbooks.eu">Music and Books</a>. Bücher und Musik. So kann man sich mit allem vollständig umgeben was einem wichtig ist, wichtiger als vieles andere auf dieser Welt. Wir betraten das Geschäft, eher ein Tempel, und der Inhaber warf uns ein Lächeln entgegen. Wir bildeten uns ein er hätte uns wiedererkannt. Wir zogen an den neu-arrangierten Regalen entlang. Ich hasse Veränderungen. Der Inhaber hat sich ein neues System für die Unmengen an Second-Hand Ware ausgedacht. Ein System das nur für ihn einen Sinn zu machen scheint. Immer wieder hörten wir seine Versuche es den Kunden zu erklären. Er scheiterte und die Kunden schienen einfach aufzugeben. Es wäre so einem Laden unwürdig gewesen, würde es von einen System verwaltet werden, welches von Jedem verstanden werden hätte können.</p> <p>Meine Beine fingen langsam an zu schmerzen. Wir gingen von einem Ende der Altstadt zum anderen Ende. Ein nicht endender Zyklus, in dem man sich zu verlieren versucht. Immer und immer wieder. Wie die vorgezeichneten meditativen Wege, die Mönche abschreiten um sich und ihre Seele zu reinigen. So vergaßen wir auf unserem Weg das Leben, das wir nun im Norden führen. Wir haben Nürnberg nie vergessen oder verlassen. Und dies gefiel mir sehr. Wir brauchten noch ein Buch um gut ausgerüstet zu sein für die <a href="http://www.roesttrommel.de/">Rösttrommel</a>. Ich habe diese Kaffeerösterei, mit angeschlossenen Cafe, erst spät entdeckt. Nicht das es nur ein Cafe mit tollem Kaffee ist, es ist die Atmosphäre, der Geruch von frisch gerösteten Kaffeebohnen.Es ist perfekt. Hier verbrachten wir teilweise Stunden über einem Buch, über Notizen, Christine neben mir am zeichnen. So wollte ich immer Leben. So etwas hatte ich vorher nicht gekannt, mir nur vorstellen können. Ich trank einen Kaffee, und noch einen, nahm Pfund für Pfund mit in unsere kleine Nürnberger Wohnung. Der Kaffee war ein Versuch diese eingebildete Kreativität mit zu nehmen, sie nicht los zulassen. So schloss ich meine verkrampften Hände um diese Vorstellung. Doch nun mussten wir wieder los. Wir kamen an all möglichen kleinen Lädchen vorbei. Alle verbunden mit kleinen Geschichten oder Erlebnissen. Der Kunsthandel in dem wir fast eine kleine Beuys Zeichnung gekauft hätten (das Wort "fast" ist der Versuch eine Traumvorstellung auszuleben, in dem man zum Kunstsammler aufgestiegen sei), das Kino in dem wir regelmäßig Filme sahen, das legendäre Copy-Land, in dem wir Christines Studienzeit verbrachten, riesige Farbabzüge machten, Zulassungsarbeitsentwürfe binden ließen, uns kaum ein Weg zu weit war um dorthin zu gelangen. Der einzige Kopierladen in dem alles zu klappen schien, Charlies Schokoladenfabrik, nur mit dem Geruch von Toner und warmen, frisch bedruckten Papier. Ja, wir wussten das immer zu schätzen. Wir gingen vorbei an Restaurants über Restaurants, viele Dates zu allen Zeitpunkten unserer Liebe.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/6194346551/"><img src="https://farm7.staticflickr.com/6155/6194346551_bbccf2900b_b.jpg" alt="img468"></a></p> <p>An dem Platz an dem sonst der Christkindlesmarkt statt findet war heute ein Austragungsort einer Bio-Messe. Der Platz war aufgefüllt mit der unweigerlichen Vermengung von Esoterik und gesundem Essen. Es ist mir ein Rätsel wieso man das Eine nicht ohne das Andere bekommt. Meine geliebten veganen Aufstriche müssen nicht aus, mit Lichtenergie angereicherten, Bärlauch hergestellt sein. Ich brauche kein Granderwasser oder Lebensmittel auf deren Einband der Barcode durchgestrichen ist. Die Barcodes, so glaubt man, seien Antennen die schlechte und bösartige Energien auf die Lebensmittel ununterbrochen abgeben. Lebensmittel für Dumme. So wie ich seither auf der Suche nach einer homöophatiefreien Apotheke bin, suche ich eine esoterik freien Biomarkt. Ich kann den Anblick der Bachblüten-Notfall-Bonbons nicht mehr ertragen. Wir gingen vorbei am Stand mit Hanf-Pesto, das von barfüssigen, mit Perlenketten behangenen und langbärtigen Männern verkauft wird. Wir kauften zwei Aufstriche und fühlten uns ein wenig dreckig dabei.</p> <p>Ein paar Minuten ins Hotel zurück. Oft fuhren wir genau diese Straße entlang. Sie lag auf dem Heimweg aus der Altstadt zurück in unsere Wohnung. Wir beobachteten die Mengen an Touristen die die Hotels in den lauen Sommerabend verließen. Bepackt mit riesigen Rucksäcken und Einkaufstüten. Diese Straße lag direkt neben dem Schienennetz, welches zum zentralen Hauptbahnhof führt. Folgt man den Schienen Richtung München, führen sie direkt an unserer alten Wohnung vorbei. Direkt unter dem Fenster unseres Schlafzimmers. Das Geräusch von wuchtig vorbeitreibenden Zügen kann so schön sein, wenn man die letzten Momente wach, die Augen gerade geschlossen, das Rattern der Waggons, vorbei ziehen hört, während man neben der Person liegt die man liebt, für die man seine Geburtsstadt verlassen hat.</p> <p>Wir hatten nur ein paar wenige Minuten zum ausruhen. Ein wenig frisches Wasser in mein Gesicht werfen bevor es weitergeht. Zwei Dinge standen noch auf unserer Liste. Wie sich herausstellte sind diese beiden Punkte im Jahr unserer Abwesenheit zu Monolithen verschmolzen. Das Burger-Restaurant <a href="http://www.premium-junkfood.de/">Auguste</a> ist, aus ihrer namensgebenden Straße, in das <a href="http://www.kunstkulturquartier.de">K4</a> gezogen.</p> <p>Vielleicht macht das K4 das Leben in Nürnberg so beneidenswert. Kino, Ausstellungsräume, Konzertsäle und mehrere offene Werkstätten. In so einer haben wir unsere ersten Siebdrucke hergestellt. Eins der Sachen die ich immer schon probieren wollte aber nie dachte die Kompetenz dafür aufbringen zu können. Hier im K4 gibt es viele nette Menschen die einen einweisen in das jeweilige Handwerks. Mehr oder weniger kommt man dann zu dem Stück das man sich erträumt hatte hier zu erschaffen. Ich stelle mir vor, dass unsere Siebe immer noch in den Schränken der Siebdruckwerkstatt auf uns warten. Das wir durch dieses Artefakt nicht vergessen wurden. Irgendwann wurden sie bestimmt gefunden,mit dem Hochdruckreiniger bearbeitet, jede Farbe aus den Sieben herausi getrieben haben, jede einzelne kleine Masche des Trägers, und die Stadt wird nie gewusst haben wer wir waren, wohin wir verschwunden sind. In einem der Konzertsäle haben wir in einer Sommernacht getanzt. Der Raum war blau erhellt, und videoprojizierte Wale schwammen die Wände entlang. In diesem Blau fühlte ich mich erstaunlich jung, weit weg und doch Zuhause, frei von Problemen, Leid und Krankheit, bildete mir ein tanzen zu können. Es schien nicht von Gewicht. Es gibt zu wenig von diesen Momenten. Momente an denen man existiert, und das auf die schönste Art und Weise. Man vergisst wer man ist und wer man gerne sein will. Alles scheint in perfekten Klang, das Licht, die Musik und die Person an seiner Seite. Nun saßen wir nebenan in diesem großartigen Burgerrestaurant und hatten kaum Hunger. Doch wir mussten einfach uns einfach hier aufhalten.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/5660258993/"><img src="https://farm6.staticflickr.com/5181/5660258993_a3601a546e_b.jpg" alt="Holga 20110427"></a></p> <p>Unser Tag folgte einer akribisch geführten Liste. Dies musste so sein. Sonst hätten die Vorhaben begraben werden können, unter alten Erinnerung und Geschichten, alles was ich mit dieser Stadt je verband. Jeder schöne Moment der in meiner Abwesenheit zu einem weiteren Einschnitt wurde. Ich kann dies überwinden, es abheilen lassen. Doch dies scheint viel Zeit zu brauchen. Mehr Zeit als Kraft oder Willen. Zulassen muss ich es irgendwann ja doch.</p> <p>Es gab da eine Band die ich zurück ließ. Es gab genau eine Probe und die Woche nach diesem Zusammentreffen kam die Entscheidung in den Norden aufzubrechen. Umso zerstörerischer hätte die Einladung zu einem Konzert dieser Band am Abend sein können. Dort würde mich eine alternative Zeitleiste auf der Bühne empfangen. Was hätte wenn, sein, können, wie. Wir trafen uns einmal in diesem Proberaum, draussen in Fürth Nord. Er lag in einem kleinen Industriegebiet. Der Größe Fürths angepasst. Eher eine Industriestraße. Ein Haus das wohl zu einem früheren Zeitpunkt ein oder zwei Firmen beheimatet hat. Nun sind dort kleine Proberäume für Bands. Der Raum, in dem ich an diesem Tag mein Equipment aufbaute, schien bei einem weiteren Gitarrenverstärker einfach aufgeben zu wollen. Aufgefüllt mir einem riesigen Schlagzeug-Set das nur eine weitere Bass-Drum und zwei kleine Tom-Toms von Neal Peart entfernt war. Es stapelten sich Synths die Wände hoch. Ich sollte heute die Percussions spielen. Auch wenn ich viel lieber Schlagzeug gespielt hätte, ich versuchte besonders experimentell zu wirken. Ich wusste sofort das diese Fassade fallen musste. Ein Pandeiro als Bassdrum, Snare, Hi-Hat und ein Ride. Trotz meinem virtuosen Kontrollverlust wollte ich nach dieser Probe weitermachen. Auch wenn ich wusste, dass die Anzahl der Tage abnehmen würde, an denen ich die Chance gehabt hätte, Teil einer Band sein zu können. Ich erinnere mich an die wunderbar stickige Sommerluft. Wir gingen an einem Sommertag und wir kamen an einem Sommertag wieder. Drei ganze Jahreszeiten ausgelassen in dieser Stadt. Als ob sie nie stattgefunden haben. Die Sommer in Nürnberg waren die schönsten meines Lebens. Nach der Probe standen wir in unseren verschwitzten T-Shirts einfach wortlos im Raum, der Nebel der Nebelmaschine schient zu erkalten aber nicht zu verfliegen, die Ohren piepsten einfach weiter und dem Druck der vergangenen zwei Stunden. Wir überlebten als Band, als Freunde, nur einen einzigen Abend, ein paar witzlose Stunden. Im Laufe der Zeit kommen immer wieder Bilder in mein Gedächnis. Es sind falsche Erinnerungen, ausgedachte Stunden mit Freunden. Es hätte alles so schön sein können.</p> <p>Die Band ließ auf sich warten und wir spazierten kopflos über die zahlosen Pegnitz-Brücken. Das Licht fing an zu dämmern und das Restblau schimmerte im Fluß. Als wir am Konzertort angekommen war, schien alles so klar. Ich war nicht mehr Teil. Ich war nie Teil. Doch nun konnte nicht mal die Illusion überleben je Teil gewesen zu sein. Am Tisch Menschen mit denen man schon einmal zuvor an Tischen saß. Niemand konnte etwas sagen. Nichts, was der Situation gerecht gewesen wäre. Und da wurde es mir bewusst. Weg mit, der über allem schwebenden, Romantik von "Hier war alles besser", nichts würde mir meine Zeit in dieser Stadt rauben können. Doch so viel liegt noch vor mir. Die Zeit mir Ihr, meiner Frau. Der wahre Grund wieso ich Nürnberg so liebe.</p>Tycho - Horizon2017-04-19T12:00:00+02:002017-04-19T12:00:00+02:00marvintag:xsteadfastx.org,2017-04-19:/2017/04/19/tycho-horizon/<p>Dieses Coachella werde ich nie verstehen. Irgendwas mit Mode die sich an Musik dranhängt um ihre Belanglosigkeit zu verschleiern. Trotzdem fallen da oft richtig gute Livesets von Bands und Musikern ab, die ich sehr mag. Aktuell die Premiere von Tychos neuen Visuells.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/EtI7f3Rwqkw' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p><p>Dieses Coachella werde ich nie verstehen. Irgendwas mit Mode die sich an Musik dranhängt um ihre Belanglosigkeit zu verschleiern. Trotzdem fallen da oft richtig gute Livesets von Bands und Musikern ab, die ich sehr mag. Aktuell die Premiere von Tychos neuen Visuells.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/EtI7f3Rwqkw' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>August Rodin und sein Höllentor2017-04-05T14:21:00+02:002017-04-05T14:21:00+02:00marvintag:xsteadfastx.org,2017-04-05:/2017/04/05/august-rodin-und-sein-hollentor/<p><a href="https://www.flickr.com/photos/marvinxsteadfast/26136279792/"><img src="https://farm2.staticflickr.com/1484/26136279792_aafc4fdf53_b.jpg" alt="Paris"></a></p> <p>Es ist schon wieder über ein Jahr her. Paris wurde mehr und mehr zu meiner Liebe. Nun verbringt man die Tage damit sich vorzustellen dort zu Leben, dabei selber aber in einer grauen Industriestadt verkommt.</p> <p>letzes Jahr sagen wir viele Museen in Paris. Eins lies mein Herz springen. Wunderschönste Melancholie …</p><p><a href="https://www.flickr.com/photos/marvinxsteadfast/26136279792/"><img src="https://farm2.staticflickr.com/1484/26136279792_aafc4fdf53_b.jpg" alt="Paris"></a></p> <p>Es ist schon wieder über ein Jahr her. Paris wurde mehr und mehr zu meiner Liebe. Nun verbringt man die Tage damit sich vorzustellen dort zu Leben, dabei selber aber in einer grauen Industriestadt verkommt.</p> <p>letzes Jahr sagen wir viele Museen in Paris. Eins lies mein Herz springen. Wunderschönste Melancholie und Kunst die mich mit Gänsehaut überzieht. Es war das Rodin Museum. Dieser bespielte Garten mitten in Paris. Die grauen Wolken damals über uns sind längst durch die Erinnerung verschwunden. Nur ein blauer Himmel blieb als wir durch den Garten liefen, immer wieder anhielten, es nicht fassen konnten.</p> <p>Rodins Todestag rundet sich und es gibt eine große Ausstellung im Grand Palais. Es ist auch der Zeitpunkt an dem Arte großartige Dokumentationen heraus haut. Zum Beispiel über sein Höllentor.</p> <iframe allowfullscreen="true" style="transition-duration:0;transition-property:no;margin:0 auto;position:relative;display:block;background-color:#000000;" frameborder="0" scrolling="no" width="100%" height="100%" src="https://www.arte.tv/player/v3/index.php?json_url=https%3A%2F%2Fapi.arte.tv%2Fapi%2Fplayer%2Fv1%2Fconfig%2Fde%2F068377-000-A%3Fautostart%3D0%26lifeCycle%3D1&amp;lang=de_DE&amp;config=arte_tvguide&amp;embed=1&amp;mute=0"></iframe>Die letzten Monate in Bildern2017-02-06T13:05:00+01:002017-02-06T13:05:00+01:00marvintag:xsteadfastx.org,2017-02-06:/2017/02/06/die-letzten-monate-in-bildern/<p><a href="https://www.flickr.com/photos/8810721@N07/31638023304/"><img alt="eisbach surfen" src="https://farm1.staticflickr.com/722/31638023304_ea0b0d49ee_b.jpg"></a> Es war mal wieder Zeit eine Rolle Film zu entwickeln. Ein wenig München und ein wenig Rügen.</p> <p><a href="https://www.flickr.com/photos/8810721@N07/32103159270/"><img alt="isar" src="https://farm1.staticflickr.com/467/32103159270_aea572c22a_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32441115586/"><img alt="sassnitz" src="https://farm1.staticflickr.com/723/32441115586_d4e0b56f88_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32481700365/"><img alt="sassnitz" src="https://farm1.staticflickr.com/455/32481700365_c31ab5af3c_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32360070841/"><img alt="sassnitz" src="https://farm1.staticflickr.com/597/32360070841_4a8d8ea878_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32481729795/"><img alt="putbus" src="https://farm1.staticflickr.com/386/32481729795_5de4e19ba7_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32103257700/"><img alt="putbus" src="https://farm1.staticflickr.com/509/32103257700_2c4739f3ec_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/31638105354/"><img alt="putbus" src="https://farm1.staticflickr.com/382/31638105354_22f7e97e00_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/31638115394/"><img alt="putbus" src="https://farm1.staticflickr.com/584/31638115394_82a63afc68_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32481776915/"><img alt="binz" src="https://farm1.staticflickr.com/417/32481776915_742476bfca_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/31638135354/"><img alt="binz" src="https://farm1.staticflickr.com/538/31638135354_0ed57162cc_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/31670214223/"><img alt="binz" src="https://farm1.staticflickr.com/353/31670214223_c81c8cd5d5_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32103318370/"><img alt="neujahr in binz" src="https://farm1.staticflickr.com/285/32103318370_631cc55aa6_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32359950051/"><img alt="münchen" src="https://farm1.staticflickr.com/693/32359950051_d2b0c295b7_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32441034736/"><img alt="eisbach surfen" src="https://farm1.staticflickr.com/770/32441034736_d6430de76e_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/31637983014/"><img alt="think coffee" src="https://farm1.staticflickr.com/388/31637983014_51566a8a9a_b.jpg"></a></p><p><a href="https://www.flickr.com/photos/8810721@N07/31638023304/"><img alt="eisbach surfen" src="https://farm1.staticflickr.com/722/31638023304_ea0b0d49ee_b.jpg"></a> Es war mal wieder Zeit eine Rolle Film zu entwickeln. Ein wenig München und ein wenig Rügen.</p> <p><a href="https://www.flickr.com/photos/8810721@N07/32103159270/"><img alt="isar" src="https://farm1.staticflickr.com/467/32103159270_aea572c22a_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32441115586/"><img alt="sassnitz" src="https://farm1.staticflickr.com/723/32441115586_d4e0b56f88_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32481700365/"><img alt="sassnitz" src="https://farm1.staticflickr.com/455/32481700365_c31ab5af3c_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32360070841/"><img alt="sassnitz" src="https://farm1.staticflickr.com/597/32360070841_4a8d8ea878_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32481729795/"><img alt="putbus" src="https://farm1.staticflickr.com/386/32481729795_5de4e19ba7_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32103257700/"><img alt="putbus" src="https://farm1.staticflickr.com/509/32103257700_2c4739f3ec_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/31638105354/"><img alt="putbus" src="https://farm1.staticflickr.com/382/31638105354_22f7e97e00_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/31638115394/"><img alt="putbus" src="https://farm1.staticflickr.com/584/31638115394_82a63afc68_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32481776915/"><img alt="binz" src="https://farm1.staticflickr.com/417/32481776915_742476bfca_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/31638135354/"><img alt="binz" src="https://farm1.staticflickr.com/538/31638135354_0ed57162cc_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/31670214223/"><img alt="binz" src="https://farm1.staticflickr.com/353/31670214223_c81c8cd5d5_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32103318370/"><img alt="neujahr in binz" src="https://farm1.staticflickr.com/285/32103318370_631cc55aa6_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32359950051/"><img alt="münchen" src="https://farm1.staticflickr.com/693/32359950051_d2b0c295b7_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/32441034736/"><img alt="eisbach surfen" src="https://farm1.staticflickr.com/770/32441034736_d6430de76e_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/31637983014/"><img alt="think coffee" src="https://farm1.staticflickr.com/388/31637983014_51566a8a9a_b.jpg"></a></p>Shoegaze Freitag: Ozean2017-01-06T11:45:00+01:002017-01-06T11:45:00+01:00marvintag:xsteadfastx.org,2017-01-06:/2017/01/06/shoegaze-freitag-ozean/<p><img alt="Ozean" src="http://xsteadfastx.org/images/ozean-ozean.jpg"></p> <p>Dank dem Hype tauchen auf einmal alte Bands auf die man selbst damals kaum kannte. Shoegaze scheint überhaupt nicht zu altern und so sind es kleine, feine Fundstücke, Schätze und Entdeckungen.</p> <p>Einer dieser Bands ist Ozean aus Los Angeles. Widerentdeckt, remastert und hochgeladen auf Bandcamp. Wunderschön.</p> <blockquote> <p>Previously unknown dream pop …</p></blockquote><p><img alt="Ozean" src="http://xsteadfastx.org/images/ozean-ozean.jpg"></p> <p>Dank dem Hype tauchen auf einmal alte Bands auf die man selbst damals kaum kannte. Shoegaze scheint überhaupt nicht zu altern und so sind es kleine, feine Fundstücke, Schätze und Entdeckungen.</p> <p>Einer dieser Bands ist Ozean aus Los Angeles. Widerentdeckt, remastert und hochgeladen auf Bandcamp. Wunderschön.</p> <blockquote> <p>Previously unknown dream pop band from USA. The demo tape was recorded in 1992 but revealed from the ocean of obscurity only less than a week ago by ex-band member Eric Shea. "UPDATE" "We just got our 1992 recordings mastered by Tim Green of Louder Studios. They're now available for download and all proceeds from will go toward funding a 2017 vinyl release: <a href="https://ozean.bandcamp.com/album/ozean">https://ozean.bandcamp.com/album/ozean</a> " Thanks! - Eric Shea</p> </blockquote> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/IvTNBbFkq4w' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Audio überall. Mopidy und Snapcast2016-12-14T13:00:00+01:002016-12-14T13:00:00+01:00marvintag:xsteadfastx.org,2016-12-14:/2016/12/14/audio-uberall-mopidy-und-snapcast/<p><a href="http://giphy.com/gifs/gilmoregirls-netflix-gilmore-girls-3ohA2ToqrPmluotB5u"><img src="http://media1.giphy.com/media/3ohA2ToqrPmluotB5u/giphy.gif" alt="source: "></a></p> <p>Musik hören geht alle Zeiten wieder durch mehrere Iterationen. Von wild, ungetaggten, Files die ich bin <a href="http://www.xmms.org/">XMMS</a> abspielte, über die ersten Versuche mit einer Verwaltung unter <a href="https://amarok.kde.org/de">Amarok</a> und eine sehr düstere Zeit mit, dem vom Herzen tiefst gehassten, iTunes. Das erste Licht am Ende des Tunnels sah ich mit <a href="https://www.musicpd.org/">MPD …</a></p><p><a href="http://giphy.com/gifs/gilmoregirls-netflix-gilmore-girls-3ohA2ToqrPmluotB5u"><img src="http://media1.giphy.com/media/3ohA2ToqrPmluotB5u/giphy.gif" alt="source: "></a></p> <p>Musik hören geht alle Zeiten wieder durch mehrere Iterationen. Von wild, ungetaggten, Files die ich bin <a href="http://www.xmms.org/">XMMS</a> abspielte, über die ersten Versuche mit einer Verwaltung unter <a href="https://amarok.kde.org/de">Amarok</a> und eine sehr düstere Zeit mit, dem vom Herzen tiefst gehassten, iTunes. Das erste Licht am Ende des Tunnels sah ich mit <a href="https://www.musicpd.org/">MPD</a> und dem Feature von <a href="https://rybczak.net/ncmpcpp/">ncmpcpp</a> Tags und Files umzubenennen und richtig zu taggen. Später dann mit <a href="https://picard.musicbrainz.org/">Picard</a> vom <a href="https://musicbrainz.org/">MusicBrainz</a> Projekt. Dann kam ich schnell auf ein Kommandozeilen-Tool was dies auch ganz wunderbar kann: <a href="http://beets.io/">beets</a>. Ab da wollte ich mir nicht mehr von anderen Programmen in meine Files schreiben lassen und erst recht nicht sortieren. Nun gibt es Clouds oder ähnliche Marketingkonstrukte und man installiert sich Software auf dem Heimserver wie <a href="https://www.plex.tv/de/">Plex</a> oder seine Open-Source-Alternative <a href="https://emby.media/">Emby</a>. Mit diesen Diensten kann man seine Mediensammlung theoretisch über das Internet konsumieren. Klappt in den besten Fällen ziemlich gut. Gerade die UI von Emby kommt mir ziemlich behäbig vor (kann natürlich auch an meinem Server liegen). Ich erinnerte mich an meine unbeschwerte Zeit mit MPD. Alles flutschte und funktionierte mit den verschiedensten Clients. Nun gibt es ein kleines Projekt mit dem Namen <a href="https://www.mopidy.com/">Mopidy</a>. Ein in Python geschriebener Audio Server der so tut als ob er MPD ist aber noch viel mehr zu bieten hat. Es gibt Plugins um auf die verschiedensten Bibliotheken zuzugreifen. Unter anderem Spotify, Youtube, Google Music, usw. Also habe ich mich mal ran gesetzt und ein <a href="https://github.com/xsteadfastx/mopidy-emby">Emby Plugin</a> geschrieben. Nun kann ich das langsame UI umgehen und viele weitere Möglichkeiten machen sich auf.</p> <p>Ich lasse Mopidy und Snapcast in einem Docker Container laufen. <a href="https://github.com/badaix/snapcast">Snapcast</a> bietet Multi-Room Streaming. Mehrere Devices die über das Netz ihren Output synchronisieren und das bei mir sogar über ein VPN hinweg. So kann ich Raspberry Pi's über all im Haus verteilen. Diese starten dann den <code>snapclient</code> nach dem booten und die Musik spielt ab.</p> <p>Mopidy muss dazu den Output in ein FIFO-File schreiben. Dazu muss in der Config folgendes stehen:</p> <div class="highlight"><pre><span></span><span class="k">[audio]</span> <span class="na">output</span> <span class="o">=</span> <span class="s">audioresample ! audioconvert ! audio/x-raw,rate=48000,channels=2,format=S16LE ! wavenc ! filesink location=/tmp/snapfifo</span> </pre></div> <p>Den Server startet man mit den Defaults mit einem einfachen <code>snapserver</code>. Der Client sollte den Server über Avahi finden. Da ich den Port an meinem Container nicht freigegeben habe mache ich das manuell mit <code>snapclient -h 192.168.1.77</code>. Mein Docker Setup findet man <a href="https://github.com/xsteadfastx/dockerfiles/tree/master/mopidy">hier</a>.</p>So Long Leonard2016-11-11T14:08:00+01:002016-11-11T14:08:00+01:00marvintag:xsteadfastx.org,2016-11-11:/2016/11/11/so-long-leonard/<p>Wenn ich Nachts, zu früh, aufwache mache ich oft Deutschlandfunk an und hoffe, dass die Monotonität der Sprache mich zurück in die Träume leitet. Schon zum zweiten mal diese Woche wache ich durch Schlagworte in den Nachrichten auf, fange im Halbschlaf nur ein paar Verben zu den Schlagworten auf, mein …</p><p>Wenn ich Nachts, zu früh, aufwache mache ich oft Deutschlandfunk an und hoffe, dass die Monotonität der Sprache mich zurück in die Träume leitet. Schon zum zweiten mal diese Woche wache ich durch Schlagworte in den Nachrichten auf, fange im Halbschlaf nur ein paar Verben zu den Schlagworten auf, mein Gehirn macht Annahmen und verbindet alles zu einem Brei, der sich wie ein Albtraum anfühlt. Vor allem wenn sich alles als wahr herausstellt. Am Mittwoch Morgen war es Donald Trump und heute Morgen war es der Name Leonard Cohen der mich, im Bett, aufschnellen ließ.</p> <p>Es ist über 10 Jahre her, als mein guter Freund Manuel mir immer wieder neue Musik zukommen ließ. All das was er als wichtig empfand, Sachen die ich kennen sollte und ohne die man höchstwahrscheinlich nicht leben könne oder ohne die man, nachdem man sie gehört hatte, nicht mehr leben konnte. Durch diese immer wachsende Sammlung fing diese sehr interessante Reise erst an. Über die frühe Folk Musik bis hin zu New Wave aus den 80er Jahren. Nun hatte ich zwei Alben von Leonard Cohen. Diese hörte ich abwechselnd mit den Alben von Bob Dylan auf meinen täglichen Touren, die ich wegen meines Zivildienstes zurück legen musste. Die Trostlosigkeit bekommt seine neue Bedeutung wenn Leonard Cohen einem vom Chelsea Hotel vorsingt.</p> <p>Es war in Israel als ich mit meinen Freunden durch die Wüste fuhr, immer wieder anhielten, uns über Musik unterhielten. Vielleicht mit die schönste Zeit meines Lebens. Eine Freundin erzählt mir von dem Leonard Cohen Konzert, das sie vor kurzem besucht hatte, wie bewegt sie war. Dieser alte Mann wie er auf dem Boden kniet, halb zerbrochen und demütig. In einem Buchladen am See Genezareth kaufte ich seinen neusten <a href="https://en.wikipedia.org/wiki/Book_of_Longing">Gedicht Band</a>. Gedichte die er in der Zeit im Zen-Kloster geschrieben hat. Er war das Thema als wir in einem Büchercafe in Jerusalem halt machten. Als uns mein Vater vom Flughafen in Deutschland abholte, hörten wir seine Lieder bis wir Zuhause eintrafen. Der Schnee umhüllte uns und wir froren, vermissten dieses sonnige und wunderschöne Land.</p> <p>Leonard Cohen spielte ein Konzert im nächsten Sommer. Und das an meinem Geburtstag. Mein Schwager und ich kauften uns die billigsten Tickets und standen da auf unseren Plätzen, Stunde um Stunde und Minute um Minute. Es war wunderschön, anmutig. Als er sang "You told me again you preferred handsome men. But for me you would make an exception" zerbrach mein Herz. Und wie die Menschen feierten als er "Berlin" besang in "First We Take Manhatten".</p> <p>Heute als ich vor Schreck aufwachte und Deutschlandfunk "Suzanne" anspielte, fiel etwas von mir ab. Ich konnte nicht mehr an mir halten... was für eine Zeit.</p> <blockquote> <p>Oh, the wind, the wind is blowing,</p> <p>through the graves the wind is blowing,</p> <p>freedom soon will come;</p> <p>then we'll come from the shadows.</p> </blockquote>My Bloody Valentine's Loveless in 5 Minuten2016-11-03T09:46:00+01:002016-11-03T09:46:00+01:00marvintag:xsteadfastx.org,2016-11-03:/2016/11/03/my-bloody-valentines-loveless-in-5-minuten/<p>Ich habe My Bloody Valentine durch den Soundtrack von Lost In Translation kennengelernt. Ich musste einfach wissen was das für eine Band ist, zu der Billy Murray und Scarlett Johansson <a href="https://www.youtube.com/watch?v=Hsj9sMDgh-A">durch das nächtliche Tokyo fahren und die Lichter der Stadt an ihnen vorbeiziehen lassen</a>. Ich bestellte mir Loveless und ich …</p><p>Ich habe My Bloody Valentine durch den Soundtrack von Lost In Translation kennengelernt. Ich musste einfach wissen was das für eine Band ist, zu der Billy Murray und Scarlett Johansson <a href="https://www.youtube.com/watch?v=Hsj9sMDgh-A">durch das nächtliche Tokyo fahren und die Lichter der Stadt an ihnen vorbeiziehen lassen</a>. Ich bestellte mir Loveless und ich brauchte zwei Durchgänge um den Zauber entfalten zu lassen. Und diese Durchgänge mussten laut sein. Sehr laut. Dann hatte mich das Album in seinen Fängen. Ich erinnere mich gut an meinen Urlaub in Jerusalem und ich Nachts alleine im Bett lag, die Wichtigkeit der gesamten Stadt auf meine Brust drückte und ich den Song <a href="https://www.youtube.com/watch?v=xdHS1sbV5xw">To Here Knows When</a> hörte, in voller Lautstärke.</p> <p>Um dieses Album ranken sich so viele Geschichten. Hier eine kleine Dokumentation dazu.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/giNSnVpclqg' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Kevin Shields über seinen Sound2016-09-30T11:49:00+02:002016-09-30T11:49:00+02:00marvintag:xsteadfastx.org,2016-09-30:/2016/09/30/kevin-shields-uber-seinen-sound/<div class="highlight"><pre><span></span>Its like a mermaid falling into a blackhole </pre></div> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/HsFZCRuA4Qs' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p><div class="highlight"><pre><span></span>Its like a mermaid falling into a blackhole </pre></div> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/HsFZCRuA4Qs' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Arcade Fire, Greta Gerwig, Spinke Jonze und überhaupt2016-09-22T10:19:00+02:002016-09-22T10:19:00+02:00marvintag:xsteadfastx.org,2016-09-22:/2016/09/22/arcade-fire-greta-gerwig-spinke-jonze-und-uberhaupt/<p>Ich habe dieses Video jetzt vielleicht 30 mal angeschaut. Auch wenn ich einige Arcade Fire Platten besitze, war mir der Hype immer zu groß. Aber wahrscheinlich bin ich wieder einmal auf dem Holzweg und schiebe super Bands vor mich hin aus unergründlichen Gründen.</p> <p>Dies ist eine Aufnahme von irgendeinem Youtube …</p><p>Ich habe dieses Video jetzt vielleicht 30 mal angeschaut. Auch wenn ich einige Arcade Fire Platten besitze, war mir der Hype immer zu groß. Aber wahrscheinlich bin ich wieder einmal auf dem Holzweg und schiebe super Bands vor mich hin aus unergründlichen Gründen.</p> <p>Dies ist eine Aufnahme von irgendeinem Youtube Event. Arcade Fire spielen live auf der Bühne und es tut sich ein ganzes Musikvideo auf. Gedreht von Spike Jonze... mit Greta Gerwig... unfassbar.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/tBTTd0gfkn0' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>semantic-ui und webpack2016-09-21T14:57:00+02:002016-09-21T14:57:00+02:00marvintag:xsteadfastx.org,2016-09-21:/2016/09/21/semantic-ui-und-webpack/<p><a href="http://giphy.com/gifs/eleventh-doctor-rR0ELLMF09bUI"><img src="http://media3.giphy.com/media/rR0ELLMF09bUI/giphy.gif" alt="source: http://www.headoverfeels.com/2013/04/25/it-does-tend-to-haunt-you-doctor-who-recap/"></a></p> <p>Hier bin ich, der mega Java Script, CSS Webentwickler. Ich habe vor vielen Jahren aufgegeben überhaupt ansatzweise CSS zu verstehen. Danke Google und viel Copy-Pasta komme ich damit auch irgendwie über die runden. Verstehen tue ich fast nichts. Noch schlimmer ist JavaScript für mich. Ich habe auch sonst nicht die …</p><p><a href="http://giphy.com/gifs/eleventh-doctor-rR0ELLMF09bUI"><img src="http://media3.giphy.com/media/rR0ELLMF09bUI/giphy.gif" alt="source: http://www.headoverfeels.com/2013/04/25/it-does-tend-to-haunt-you-doctor-who-recap/"></a></p> <p>Hier bin ich, der mega Java Script, CSS Webentwickler. Ich habe vor vielen Jahren aufgegeben überhaupt ansatzweise CSS zu verstehen. Danke Google und viel Copy-Pasta komme ich damit auch irgendwie über die runden. Verstehen tue ich fast nichts. Noch schlimmer ist JavaScript für mich. Ich habe auch sonst nicht die riesige Programmiererfahrung und fühle mich in meiner kleinen Python-Welt ziemlich wohl. Es gibt aber Projekte an denen kommt man nicht darum herum ein wenig Webkram zu erledigen.</p> <p>Ich habe mich dafür entschieden diese kleine Firmenseite mit <a href="https://www.getlektor.com">Lektor</a> zu bauen. Ein CMS, gebaut in Python, das statisches HTML generiert und wahnsinnig flexibel mit den Datenstrukturen ist. Man kann Datenmodele anlegen für bestimmte Arten von Seiten die wiederum vorgefertigte Elemente besitzen. Ich habe eine weile gebraucht und bin zum Fan geworden. Und stimmten die Datenmodelle kommt man schnell an den Punkt an dem es keinen Spaß macht: Die Templates. Versteht mich nicht falsch, die Templatesprache <a href="http://jinja.pocoo.org/docs/dev/">Jinja2</a> ganz wunderbar. Es ist HTML, CSS und JavaScript das mich regelmäßig gehirnfurzen läßt. Welche Frameworks setzt man ein? Im Endeffekt ist man beim erstem import schon veraltet und verpasst immer den letzten heißen Scheiß. Die Seite wanderte von <a href="http://purecss.io/">purecss</a> über <a href="http://getbootstrap.com/">Bootstrap</a> zu <a href="http://semantic-ui.com/">Semantic UI</a>. Nun habe ich in den Docs von Lektor von <a href="https://www.getlektor.com/docs/guides/webpack/">webpack</a> erfahren. Die Beschreibung auf der <a href="https://webpack.github.io/">offizielle Website</a> hat mich noch mehr in die Verzweifelung getrieben. Aber wenn ich es nun richtig verstanden habe generiert er statische Assets und kann die Files kleiner machen. Man wirft ihm sein JavaScript (mit allen Abhängikeiten) und sein CSS (mit allen Abhängigkeiten) hin und er generiert ein JS und ein CSS daraus. Dies funktioniert mit manchen Frameworks besser und mit manchen weniger. Semantic-UI scheint da so seine <a href="https://github.com/Semantic-Org/Semantic-UI/issues/3533">Probleme</a> zu haben. Hier also der Weg der für mich funktioniert hat.</p> <p>Meine <code>webpack.config.js</code>:</p> <div class="highlight"><pre><span></span><span class="kd">var</span> <span class="nx">webpack</span> <span class="o">=</span> <span class="nx">require</span><span class="p">(</span><span class="s1">&#39;webpack&#39;</span><span class="p">);</span> <span class="kd">var</span> <span class="nx">path</span> <span class="o">=</span> <span class="nx">require</span><span class="p">(</span><span class="s1">&#39;path&#39;</span><span class="p">);</span> <span class="kd">var</span> <span class="nx">ExtractTextPlugin</span> <span class="o">=</span> <span class="nx">require</span><span class="p">(</span><span class="s1">&#39;extract-text-webpack-plugin&#39;</span><span class="p">);</span> <span class="kd">var</span> <span class="nx">options</span> <span class="o">=</span> <span class="p">{</span> <span class="nx">entry</span><span class="o">:</span> <span class="p">{</span> <span class="s1">&#39;app&#39;</span><span class="o">:</span> <span class="s1">&#39;./js/main.js&#39;</span><span class="p">,</span> <span class="s1">&#39;styles&#39;</span><span class="o">:</span> <span class="s1">&#39;./scss/main.scss&#39;</span> <span class="p">},</span> <span class="nx">output</span><span class="o">:</span> <span class="p">{</span> <span class="nx">path</span><span class="o">:</span> <span class="nx">path</span><span class="p">.</span><span class="nx">dirname</span><span class="p">(</span><span class="nx">__dirname</span><span class="p">)</span> <span class="o">+</span> <span class="s1">&#39;/assets/static/gen&#39;</span><span class="p">,</span> <span class="nx">filename</span><span class="o">:</span> <span class="s1">&#39;[name].js&#39;</span> <span class="p">},</span> <span class="nx">devtool</span><span class="o">:</span> <span class="s1">&#39;#cheap-module-source-map&#39;</span><span class="p">,</span> <span class="nx">resolve</span><span class="o">:</span> <span class="p">{</span> <span class="nx">modulesDirectories</span><span class="o">:</span> <span class="p">[</span><span class="s1">&#39;node_modules&#39;</span><span class="p">],</span> <span class="nx">extensions</span><span class="o">:</span> <span class="p">[</span><span class="s1">&#39;&#39;</span><span class="p">,</span> <span class="s1">&#39;.js&#39;</span><span class="p">]</span> <span class="p">},</span> <span class="nx">module</span><span class="o">:</span> <span class="p">{</span> <span class="nx">loaders</span><span class="o">:</span> <span class="p">[</span> <span class="p">{</span> <span class="nx">test</span><span class="o">:</span> <span class="sr">/\.js$/</span><span class="p">,</span> <span class="nx">exclude</span><span class="o">:</span> <span class="sr">/node_modules/</span><span class="p">,</span> <span class="nx">loader</span><span class="o">:</span> <span class="s1">&#39;babel-loader&#39;</span><span class="p">,</span> <span class="nx">query</span><span class="o">:</span> <span class="p">{</span> <span class="nx">presets</span><span class="o">:</span> <span class="p">[</span><span class="s1">&#39;es2015&#39;</span><span class="p">],</span> <span class="p">}</span> <span class="p">},</span> <span class="p">{</span> <span class="nx">test</span><span class="o">:</span> <span class="sr">/\.scss$/</span><span class="p">,</span> <span class="nx">loader</span><span class="o">:</span> <span class="nx">ExtractTextPlugin</span><span class="p">.</span><span class="nx">extract</span><span class="p">(</span><span class="s1">&#39;style-loader&#39;</span><span class="p">,</span> <span class="s1">&#39;css-loader!sass-loader&#39;</span><span class="p">)</span> <span class="p">},</span> <span class="p">{</span> <span class="nx">test</span><span class="o">:</span> <span class="sr">/\.css$/</span><span class="p">,</span> <span class="nx">loader</span><span class="o">:</span> <span class="nx">ExtractTextPlugin</span><span class="p">.</span><span class="nx">extract</span><span class="p">(</span><span class="s1">&#39;style-loader&#39;</span><span class="p">,</span> <span class="s1">&#39;css-loader&#39;</span><span class="p">)</span> <span class="p">},</span> <span class="p">{</span> <span class="nx">test</span><span class="o">:</span> <span class="sr">/\.(woff2?|ttf|eot|svg|png|gif)(\?.*?)?$/</span><span class="p">,</span> <span class="nx">loader</span><span class="o">:</span> <span class="s1">&#39;file&#39;</span> <span class="p">}</span> <span class="p">]</span> <span class="p">},</span> <span class="nx">plugins</span><span class="o">:</span> <span class="p">[</span> <span class="k">new</span> <span class="nx">webpack</span><span class="p">.</span><span class="nx">optimize</span><span class="p">.</span><span class="nx">UglifyJsPlugin</span><span class="p">({</span> <span class="nx">compress</span><span class="o">:</span> <span class="p">{</span> <span class="nx">warnings</span><span class="o">:</span> <span class="kc">false</span> <span class="p">}</span> <span class="p">}),</span> <span class="k">new</span> <span class="nx">webpack</span><span class="p">.</span><span class="nx">optimize</span><span class="p">.</span><span class="nx">DedupePlugin</span><span class="p">(),</span> <span class="k">new</span> <span class="nx">webpack</span><span class="p">.</span><span class="nx">optimize</span><span class="p">.</span><span class="nx">OccurenceOrderPlugin</span><span class="p">(),</span> <span class="k">new</span> <span class="nx">webpack</span><span class="p">.</span><span class="nx">ProvidePlugin</span><span class="p">({</span> <span class="nx">$</span><span class="o">:</span> <span class="s1">&#39;jquery&#39;</span><span class="p">,</span> <span class="nx">jQuery</span><span class="o">:</span> <span class="s1">&#39;jquery&#39;</span> <span class="p">}),</span> <span class="k">new</span> <span class="nx">ExtractTextPlugin</span><span class="p">(</span><span class="s1">&#39;styles.css&#39;</span><span class="p">,</span> <span class="p">{</span> <span class="nx">allChunks</span><span class="o">:</span> <span class="kc">true</span> <span class="p">})</span> <span class="p">]</span> <span class="p">};</span> <span class="nx">module</span><span class="p">.</span><span class="nx">exports</span> <span class="o">=</span> <span class="nx">options</span><span class="p">;</span> </pre></div> <p>Ich habe alle Komponenten die ich von Semantic-UI brauche separat installiert:</p> <p><code>npm install -save-dev semantic-ui-css semantic-ui-sidebar semantic-ui-transition semantic-ui-visibility</code></p> <p>Dann habe ich alle CSS files in meine <code>/scss/main.scss</code> importiert:</p> <div class="highlight"><pre><span></span><span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/reset.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/site.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/container.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/grid.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/header.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/image.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/menu.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/divider.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/dropdown.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/segment.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/button.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/list.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/icon.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/sidebar.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~semantic-ui-css/components/transition.css&quot;</span><span class="p">;</span> <span class="cp">@import</span> <span class="s2">&quot;~slick-carousel/slick/slick.css&quot;</span><span class="p">;</span> </pre></div> <p>Alles anderen Versuche dies über meine <code>/js/main.js</code> zu machen, gelangen mir nicht. Alle möglichen <code>require('semantic-ui-css/components/reset.css')</code> wollten es einfach nicht tun. Als <code>@import</code> in der <code>/scss/main.scss</code> war es kein Problem. Nun kommen wir zum JavaScript. Normalerweise reicht ein <code>require('meinetollejslib')</code> und webpack hangelt sich durch die files und macht damit Sachen. Nur mit den Komponenten von Sematic wollte es nicht wirklich. Woran es genau lag weiß ich auch nicht so recht. Hat wohl was mit der Mondphase und rechtsdrehenden Joghurts zu tun. Die Lösung fand ich dann doch in einem <a href="https://github.com/Semantic-Org/Semantic-UI/issues/3533">Bugreport</a>. Hier meine <code>/js/main.js</code>:</p> <div class="highlight"><pre><span></span><span class="kn">import</span> <span class="err">$</span> <span class="kn">from</span> <span class="s1">&#39;jquery&#39;</span> <span class="err">$</span><span class="o">.</span><span class="n">fn</span><span class="o">.</span><span class="n">sidebar</span> <span class="o">=</span> <span class="n">require</span><span class="p">(</span><span class="s1">&#39;semantic-ui-sidebar&#39;</span><span class="p">)</span> <span class="err">$</span><span class="o">.</span><span class="n">fn</span><span class="o">.</span><span class="n">transition</span> <span class="o">=</span> <span class="n">require</span><span class="p">(</span><span class="s1">&#39;semantic-ui-transition&#39;</span><span class="p">)</span> <span class="err">$</span><span class="o">.</span><span class="n">fn</span><span class="o">.</span><span class="n">visibility</span><span class="o">=</span> <span class="n">require</span><span class="p">(</span><span class="s1">&#39;semantic-ui-visibility&#39;</span><span class="p">)</span> </pre></div> <p>Auf einmal kann ich die jquery Funtionen nutzen. Hexenwerk.</p>Moby und das Schlafen2016-09-19T12:09:00+02:002016-09-19T12:09:00+02:00marvintag:xsteadfastx.org,2016-09-19:/2016/09/19/moby-und-das-schlafen/<p><img alt="Moby" src="http://xsteadfastx.org/images/moby-calm_sleep.jpg"></p> <p>Moby hat ein 4 Stündiges Ambient Werk freigegeben. Anscheinend hat Moby seit langem ein Problem mit dem schlafen. Diese Tracks sind für ihn eine Art Versuch zu Ruhe zu kommen.</p> <blockquote> <p>over the last couple of years i’ve been making really really really quiet music to listen to when i …</p></blockquote><p><img alt="Moby" src="http://xsteadfastx.org/images/moby-calm_sleep.jpg"></p> <p>Moby hat ein 4 Stündiges Ambient Werk freigegeben. Anscheinend hat Moby seit langem ein Problem mit dem schlafen. Diese Tracks sind für ihn eine Art Versuch zu Ruhe zu kommen.</p> <blockquote> <p>over the last couple of years i’ve been making really really really quiet music to listen to when i do yoga or sleep or meditate or panic. i ended up with 4 hours of music and have decided to give it away. you can download it for free here: <a href="moby.com/la1">moby.com/la1</a> or stream it for free on soundcloud. or you can get it on spotify, apple music &amp; itunes &amp; deezer &amp; tidal. it’s really quiet: no drums, no vocals, just very slow calm pretty chords and sounds and things for sleeping and yoga and etc. and feel free to share it or give it away or whatever, it’s not protected</p> </blockquote> <p><iframe width="100%" height="450" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?visual=true&url=http%3A%2F%2Fapi.soundcloud.com%2Fplaylists%2F217337348&show_artwork=true&lf=c1ce5c176a7fdd95952f3462b27b4c9f"></iframe></p> <p>(<a href="http://www.openculture.com/2016/06/moby-lets-you-download-4-hours-of-ambient-music-to-help-you-sleep.html">via</a>)</p>Die Inspiration zu "Before Sunrise"2016-09-14T12:15:00+02:002016-09-14T12:15:00+02:00marvintag:xsteadfastx.org,2016-09-14:/2016/09/14/die-inspiration-zu-before-sunrise/<p>Dieser Film. Jeder der mich und meine Biographie kennt weiß über die Verbindung zu diesem Film. Ich hatte mal gehört das es eine echte Nacht gab die Richard Linklater inspirierte. Es war Amy Lehrhaupt mit der er durch die Nacht zog. Aber ich will nicht spoilern...</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/zb0Q7QyvrHM' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p><p>Dieser Film. Jeder der mich und meine Biographie kennt weiß über die Verbindung zu diesem Film. Ich hatte mal gehört das es eine echte Nacht gab die Richard Linklater inspirierte. Es war Amy Lehrhaupt mit der er durch die Nacht zog. Aber ich will nicht spoilern...</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/zb0Q7QyvrHM' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Alien - H. R. Giger's Beautiful Monster2016-09-14T09:36:00+02:002016-09-14T09:36:00+02:00marvintag:xsteadfastx.org,2016-09-14:/2016/09/14/alien-h-r-gigers-beautiful-monster/<p>Der Youtuber <a href="https://www.youtube.com/channel/UCuPgdqQKpq4T4zeqmTelnFg">kaptainkristian</a> hat ein kleines Video online gestellt in dem es um die Schönheit des Aliens geht. Ich habe oft so meine Probleme mit den Arbeiten von H.R. Giger aber Alien ist ein Meisterwerk. Schade das es nie zu einem Dune Film kam mit seinen Werken. Ich habe …</p><p>Der Youtuber <a href="https://www.youtube.com/channel/UCuPgdqQKpq4T4zeqmTelnFg">kaptainkristian</a> hat ein kleines Video online gestellt in dem es um die Schönheit des Aliens geht. Ich habe oft so meine Probleme mit den Arbeiten von H.R. Giger aber Alien ist ein Meisterwerk. Schade das es nie zu einem Dune Film kam mit seinen Werken. Ich habe viel Bewunderung für seine Arbeiten, auch wenn sie mich ganz schön abfucken.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/q4Zzag7MusM' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Ulrich Schnauss und Tangerine Dream covern Stranger Things Soundtrack2016-09-13T09:57:00+02:002016-09-13T09:57:00+02:00marvintag:xsteadfastx.org,2016-09-13:/2016/09/13/ulrich-schnauss-und-tangerine-dream-covern-stranger-things-soundtrack/<p>Ich vertrete mal eine unpopuläre Meinung. Ich mochte Stranger Things eigentlich fast nur wegen dem Soundtrack. Inhaltlich hat es mich nicht so überzeugt, auch wenn ich die jungen Schauspieler wirklich gut fand. Geschaut habe ich die Serie dann doch bis zum Ende. Man hofft das es noch irgendeinen Twist gibt …</p><p>Ich vertrete mal eine unpopuläre Meinung. Ich mochte Stranger Things eigentlich fast nur wegen dem Soundtrack. Inhaltlich hat es mich nicht so überzeugt, auch wenn ich die jungen Schauspieler wirklich gut fand. Geschaut habe ich die Serie dann doch bis zum Ende. Man hofft das es noch irgendeinen Twist gibt der einen doch umhaut.</p> <p>Fast ging es an mir vorbei: Ulrich Schnauss ist ja jetzt festes Mitglied bei Tangerine Dream. Und die haben nun zusammen einen Coversong eines Tracks aus Stranger Things aufgenommen. So gut der richtige Soundtrack ist, ich glaube ich würde sofort Ulrich Schnauss nehmen um den Soundtrack zu schreiben. Egal für welchen Film...</p> <p><strong>Update:</strong> Scheint nicht mehr online zu sein... meh.</p>American Football - Give Me The Gun2016-09-08T10:21:00+02:002016-09-08T10:21:00+02:00marvintag:xsteadfastx.org,2016-09-08:/2016/09/08/american-football-give-me-the-gun/<p>Hier also die zweite Single aus dem neuen American Football Album. Ich brauche noch ein wenig...</p> <p><iframe width="100%" height="400" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?visual=true&url=http%3A%2F%2Fapi.soundcloud.com%2Ftracks%2F281657112&show_artwork=true"></iframe></p><p>Hier also die zweite Single aus dem neuen American Football Album. Ich brauche noch ein wenig...</p> <p><iframe width="100%" height="400" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?visual=true&url=http%3A%2F%2Fapi.soundcloud.com%2Ftracks%2F281657112&show_artwork=true"></iframe></p>Auf den Dächern der Stadt: Paris2016-09-07T12:28:00+02:002016-09-07T12:28:00+02:00marvintag:xsteadfastx.org,2016-09-07:/2016/09/07/auf-den-dachern-der-stadt-paris/<p>Wenn man mal wieder Fernweh braucht. Arte sendet gerade eine Reihe mit dem Namen "Auf den Dächern". Diese <a href="http://www.arte.tv/guide/de/058364-001-A/auf-den-daechern-der-stadt">Episode</a> handelt von dem wunderschönen Paris... und mal ehrlich... Gärten auf den Dächern Paris... pure Liebe.</p> <iframe src="http://www.arte.tv/guide/de/embed/058364-001-A/medium" allowfullscreen="true" style="width: 600px; height: 344px;" frameborder="0"></iframe><p>Wenn man mal wieder Fernweh braucht. Arte sendet gerade eine Reihe mit dem Namen "Auf den Dächern". Diese <a href="http://www.arte.tv/guide/de/058364-001-A/auf-den-daechern-der-stadt">Episode</a> handelt von dem wunderschönen Paris... und mal ehrlich... Gärten auf den Dächern Paris... pure Liebe.</p> <iframe src="http://www.arte.tv/guide/de/embed/058364-001-A/medium" allowfullscreen="true" style="width: 600px; height: 344px;" frameborder="0"></iframe>Yumi Zouma - Keep It Close To Me2016-09-02T08:30:00+02:002016-09-02T08:30:00+02:00marvintag:xsteadfastx.org,2016-09-02:/2016/09/02/yumi-zouma-keep-it-close-to-me/<p>Eigentlich denkt man sich, dass Dreampop langsam den Shark gejumped hat. Aber dann freut man sich einfach über dahintröpfelnde Beats mit Synthiteppich. Heute ist so ein Tag und da passt sich, dass Yumi Zouma gestern ihr erstes Video online gestellt hat.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/gZusfpGiIkQ' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p><p>Eigentlich denkt man sich, dass Dreampop langsam den Shark gejumped hat. Aber dann freut man sich einfach über dahintröpfelnde Beats mit Synthiteppich. Heute ist so ein Tag und da passt sich, dass Yumi Zouma gestern ihr erstes Video online gestellt hat.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/gZusfpGiIkQ' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Die ersten Bilder mit der Olympus MJU-12016-09-01T10:14:00+02:002016-09-01T10:14:00+02:00marvintag:xsteadfastx.org,2016-09-01:/2016/09/01/die-ersten-bilder-mit-der-olympus-mju-1/<p><a href="https://www.flickr.com/photos/8810721@N07/29298872806/"><img alt="kerksiek" src="https://farm9.staticflickr.com/8808/29298872806_fc1eb3c546_b.jpg"></a></p> <p>Das mit dem Fotos entwickeln ist schon so eine Sache. Will ich mir besonders Mühe geben, greife ich doch lieber zu einer richtigen Kamera anstatt zu meinem Telefon. Da dauert es aber maximal 35 weitere Fotos bis ich den Film entwickeln, scannen und aufarbeiten kann. So kann es es sein …</p><p><a href="https://www.flickr.com/photos/8810721@N07/29298872806/"><img alt="kerksiek" src="https://farm9.staticflickr.com/8808/29298872806_fc1eb3c546_b.jpg"></a></p> <p>Das mit dem Fotos entwickeln ist schon so eine Sache. Will ich mir besonders Mühe geben, greife ich doch lieber zu einer richtigen Kamera anstatt zu meinem Telefon. Da dauert es aber maximal 35 weitere Fotos bis ich den Film entwickeln, scannen und aufarbeiten kann. So kann es es sein das ich durch ein paar uninspierirte Wochen und Monate wandel ohne weiter zu fotografieren. Der Film ist dabei weiterhin in einer Art Dornröschenschlaf und wartet nur darauf durch ein Bad in <a href="https://de.wikipedia.org/wiki/Rodinal">Rodinal</a> erwacht zu werden. So war es auch mit dem Film den ich in meiner "neuen" Olympus MJU-1 hatte. Mehrere Monate Tiefschlaf. Hier eine kleine Zusammenfassung nach dem Aufwachen.</p> <p><a href="https://www.flickr.com/photos/8810721@N07/28740829083/"><img alt="2016-08-31-0007" src="https://farm9.staticflickr.com/8328/28740829083_b077be85c1_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/28738796264/"><img alt="2016-08-31-0005" src="https://farm9.staticflickr.com/8348/28738796264_be1c1924cc_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/29328537986/"><img alt="2016-08-31-0010" src="https://farm9.staticflickr.com/8280/29328537986_a1d6466e83_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/28740812573/"><img alt="2016-08-31-0008" src="https://farm9.staticflickr.com/8855/28740812573_3b12d7afa9_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/29283407201/"><img alt="2016-08-31-0006" src="https://farm9.staticflickr.com/8445/29283407201_eac4bde409_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/28740895113/"><img alt="2016-08-31-0002" src="https://farm9.staticflickr.com/8111/28740895113_6e222ba16f_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/28740914833/"><img alt="2016-08-31-0001" src="https://farm9.staticflickr.com/8788/28740914833_cf1c999c65_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/29224414062/"><img alt="eli" src="https://farm9.staticflickr.com/8185/29224414062_50472fa1ef_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/29045124040/"><img alt="eli" src="https://farm9.staticflickr.com/8250/29045124040_06fd8b8d06_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/29044993530/"><img alt="klein brunsrode" src="https://farm9.staticflickr.com/8687/29044993530_52c4e6da45_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/29298844816/"><img alt="klein brunsrode" src="https://farm9.staticflickr.com/8269/29298844816_f98b3b3532_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/29045002150/"><img alt="2016-08-29-0008" src="https://farm9.staticflickr.com/8771/29045002150_4a5ff150f2_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/29298855156/"><img alt="umberto" src="https://farm9.staticflickr.com/8344/29298855156_28cdab616a_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/29045061470/"><img alt="klein brunsrode" src="https://farm9.staticflickr.com/8410/29045061470_5df300cd9f_b.jpg"></a></p>American Football - I've Been So Lost For So Long2016-08-24T10:15:00+02:002016-08-24T10:15:00+02:00marvintag:xsteadfastx.org,2016-08-24:/2016/08/24/american-football-ive-been-so-lost-for-so-long/<p><img alt="I've Been So Lost For So Long" src="http://xsteadfastx.org/images/american_football-ive_been_so_lost_for_so_long.jpg"> Es war zu erwarten das nach der großen Reunion auch neue Songs kommen. Das Album "American Football" stand für sich. Ein Meisterwerk was erst wenig Furore machte und doch von vielen geliebt wurde. Ich habe es im Jahr 2001 entdeckt als ich Livejournal Emo Blogs nach neuen Bands absuchte und …</p><p><img alt="I've Been So Lost For So Long" src="http://xsteadfastx.org/images/american_football-ive_been_so_lost_for_so_long.jpg"> Es war zu erwarten das nach der großen Reunion auch neue Songs kommen. Das Album "American Football" stand für sich. Ein Meisterwerk was erst wenig Furore machte und doch von vielen geliebt wurde. Ich habe es im Jahr 2001 entdeckt als ich Livejournal Emo Blogs nach neuen Bands absuchte und im Internet hinterher jagte. Das Album ist mit mir durch mein halbes Leben gegangen. Nun also gibt es einen ersten neuen Song. Ich mag die Weiterführung mit dem Cover, das das Innere des Hauses vom ersten Album sein könnte. Ich hoffe einfach das es so ist. Ich bin kritisch aber ich werde mir das Album kaufen und immer und immer wieder hören. Vielleicht der Soundtrack für die nächsten Lebensabschnitte.</p> <p><iframe width="100%" height="400" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?visual=true&url=http%3A%2F%2Fapi.soundcloud.com%2Ftracks%2F278986607&show_artwork=true"></iframe></p>Shoegaze Freitag: Jaguwar - Eyes Collide2016-08-19T12:35:00+02:002016-08-19T12:35:00+02:00marvintag:xsteadfastx.org,2016-08-19:/2016/08/19/shoegaze-freitag-jaguwar-eyes-collide/<p><img alt="Jaguwar" src="http://xsteadfastx.org/images/jaguwar-eyescollide.jpg"></p> <p>Jaguwar kommen aus Dresden und habe mit diesem Track einen riesigen Shoegaze Song geschaffen. Mit dem Kopf durch die Soundwand.</p> <p><iframe width="100%" height="400" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?visual=true&url=http%3A%2F%2Fapi.soundcloud.com%2Ftracks%2F256801702&show_artwork=true"></iframe></p><p><img alt="Jaguwar" src="http://xsteadfastx.org/images/jaguwar-eyescollide.jpg"></p> <p>Jaguwar kommen aus Dresden und habe mit diesem Track einen riesigen Shoegaze Song geschaffen. Mit dem Kopf durch die Soundwand.</p> <p><iframe width="100%" height="400" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?visual=true&url=http%3A%2F%2Fapi.soundcloud.com%2Ftracks%2F256801702&show_artwork=true"></iframe></p>Tycho - Division2016-08-17T09:47:00+02:002016-08-17T09:47:00+02:00marvintag:xsteadfastx.org,2016-08-17:/2016/08/17/tycho-division/<p><img alt="Division" src="http://xsteadfastx.org/images/tycho-division.jpg"></p> <p>Das Richtige für einen Morgen an den man durch den herbstlichen Morgennebel geradelt ist, auch wenn es eigentlich die Mitte des Sommers sein sollte. Ein neuer Tycho Track. Toll!</p> <p><iframe width="100%" height="400" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?visual=true&url=http%3A%2F%2Fapi.soundcloud.com%2Ftracks%2F271657161&show_artwork=true"></iframe></p><p><img alt="Division" src="http://xsteadfastx.org/images/tycho-division.jpg"></p> <p>Das Richtige für einen Morgen an den man durch den herbstlichen Morgennebel geradelt ist, auch wenn es eigentlich die Mitte des Sommers sein sollte. Ein neuer Tycho Track. Toll!</p> <p><iframe width="100%" height="400" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?visual=true&url=http%3A%2F%2Fapi.soundcloud.com%2Ftracks%2F271657161&show_artwork=true"></iframe></p>Eine Liebeserklärung an Mistress America2016-08-12T13:43:00+02:002016-08-12T13:43:00+02:00marvintag:xsteadfastx.org,2016-08-12:/2016/08/12/eine-liebeserklarung-an-mistress-america/<p><img alt="Mistress America" src="http://xsteadfastx.org/images/mistress-america.jpg"></p> <p>Wenn ich an Frances Ha denke, muss ich an die Szene denken wie, alleine in Paris, am Seine Ufer langgeht. Im Hintergrund der Eiffelturm in seiner Funktion als Leuchtturm, in den von Lichtern erstrahlten Paris.</p> <p>Ich habe ein Herz für verträumte Filme, diese Atmosphäre die einen Plot fast nicht vermissen …</p><p><img alt="Mistress America" src="http://xsteadfastx.org/images/mistress-america.jpg"></p> <p>Wenn ich an Frances Ha denke, muss ich an die Szene denken wie, alleine in Paris, am Seine Ufer langgeht. Im Hintergrund der Eiffelturm in seiner Funktion als Leuchtturm, in den von Lichtern erstrahlten Paris.</p> <p>Ich habe ein Herz für verträumte Filme, diese Atmosphäre die einen Plot fast nicht vermissen lassen. Stimmt dies, die Farben, die Stimmung, braucht nicht mehr viel passieren und es hat mich. Klar passieren Sachen in The Virgin Suicides oder Somewhere, doch es ist nicht die Geschichte die mich diese Filme wieder und immer wieder schauen lassen. Ich lege sie in bestimmten Situationen ein, immer dann wenn ich eine Inspiration suche, etwas das meine Melancholie unterstreicht. Und wenn ich nicht Zuhause bin, muss sichergestellt sein, dass ich zumindestens den Soundtrack in irgendeiner Form dabei habe. Ein kleiner Notanker.</p> <p>Vor ein paar Jahren habe ich den Film "Der Tintenfisch und der Wal" und später "Greenberg" angeschaut. Es sind diese Tragikkomödien mit wunderbarer Kamerarbeit, Farben und Soundtrack. Und dann kam Frances Ha. Ich beschreibe ihn immer als die perfekte Darstellung der Sinnsuche unserer Generation. Die Generation mit allen Möglichkeiten und dem lebensbestimmenden Wunsch alles erlebt und bereist zu haben. Nur dies würde ein erfolgreiches und erfülltes Leben bedeuten. Am Ende schwimmt man nur zwischen gescheiterten Projekten und Träumen umher und das an einem sehr uncoolen Ort. Frances (gespielt von Greta Gerwig) weiß auch nicht so wirklich was sie tun soll, wie es weitergeht mit diesem Erwachsensein. Das ganze gefilmt in wunderschönen Schwarz-Weiß.</p> <p>Nun habe ich vor ein paar Tagen "Mistress America" gesehen. Geschrieben von Noah Baumbach und Greta Gerwig. Der Film fühlt sich an als ob er im gleichen Universum wie Frances Ha spielt. Greta Gerwig spielt einen ähnlichen Charakter wie Frances, ohne das Ende des eigenen Filmes erlebt zu haben. Dort sind immer noch die Projekte und Träume, die vielen unausgeführten Ideen, das Problem Erwachsen zu sein. In diesem Fall Brooke, wird von ihrer Schwester in spe, Tracy in New York aufgesucht. Diese ist gerade 18 und studiert irgendwas mit Literatur und findet nicht wirklich Anschluss an das Studentenleben am College. Sie hängt sich an das Leben von Brooke, die sie durch ihre Tragik, ihren Humor, ihrer Lebensweisheit, zu einer Kurzgeschichte mit dem Namen "Mistress America" inspiriert. Mehr will ich gar nicht dazu sagen. Es ist diese melancholische Atmosphäre, der Soundtrack und vor allem der Humor. Großartig.</p> <p>Und ja der Soundtrack, er ist perfekt. Dean Wareham und Britta Philips haben ihre Hausaufgaben gemacht. Es klingt so als ob New Order Dreampop machen würden, durch und durch. Und mit diesen Songs gleitet man weiter durch das Erwachsensein, mit all den unerfüllten Ideen, an immer den falschen Orten sein, an denen man zu wenig erlebt und muss sich die Fotos der Reisen der "Freunde" auf Facebook anschauen. Denn das ist was uns ausmacht.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/6z8MCW16uZY' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Hey Mercedes Reunion in 360°2016-08-01T13:00:00+02:002016-08-01T13:00:00+02:00marvintag:xsteadfastx.org,2016-08-01:/2016/08/01/hey-mercedes-reunion-in-360/<p>Hey Mercedes hätten sich damals bestimmt vieles vorstellen können zu ihrem 15 Jährigen Jubiläum. Wo sind die Raketenrucksäcke? Nun spielten sie in Brooklyn eine Show und die wurde per 360° Kamera aufgezeichnet. Braucht man das? Bestimmt nicht. Die Musik steht für sich. Cool ist es aber schon...</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/RC8242JeRwk' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/gRI7fLruKpU' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/U0-jXiPFVNs' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/71Ug0sPChU4' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/99ittTLtMss' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/b7Kb-4AWS3o' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/-R3-eY8yX1I' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/NXN5FfkDVug' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/SgktHdY9DEQ' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p><p>Hey Mercedes hätten sich damals bestimmt vieles vorstellen können zu ihrem 15 Jährigen Jubiläum. Wo sind die Raketenrucksäcke? Nun spielten sie in Brooklyn eine Show und die wurde per 360° Kamera aufgezeichnet. Braucht man das? Bestimmt nicht. Die Musik steht für sich. Cool ist es aber schon...</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/RC8242JeRwk' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/gRI7fLruKpU' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/U0-jXiPFVNs' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/71Ug0sPChU4' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/99ittTLtMss' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/b7Kb-4AWS3o' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/-R3-eY8yX1I' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/NXN5FfkDVug' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/SgktHdY9DEQ' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>xonsh aus Bash heraus starten2016-07-29T14:39:00+02:002016-07-29T14:39:00+02:00marvintag:xsteadfastx.org,2016-07-29:/2016/07/29/xonsh-aus-bash-heraus-starten/<p>Noch ein kleiner Nachtrag zu dem <a href="http://xsteadfastx.org/2016/07/28/meine-neue-shell-xonsh/">Artikel</a> von gestern. Heute wollte ich mal wieder in meiner Blogging Vagrant Docker Box alles auf den neusten stand bringen. Ich benutze dazu <code>vagrant provision</code>. Dies startet das Provisioning, also holt meine neuen Config-Files. installiert neue Software usw. Ansible benutzt SSH um auf die …</p><p>Noch ein kleiner Nachtrag zu dem <a href="http://xsteadfastx.org/2016/07/28/meine-neue-shell-xonsh/">Artikel</a> von gestern. Heute wollte ich mal wieder in meiner Blogging Vagrant Docker Box alles auf den neusten stand bringen. Ich benutze dazu <code>vagrant provision</code>. Dies startet das Provisioning, also holt meine neuen Config-Files. installiert neue Software usw. Ansible benutzt SSH um auf die Vagrant Box zuzugreifen und dies war auf einmal ein Problem. Ansible lief sofort an die Wand.</p> <div class="highlight"><pre><span></span><span class="o">/</span><span class="nt">bin</span><span class="o">/</span><span class="nt">sh</span><span class="o">:</span> <span class="nt">sudo</span> <span class="nt">-H</span> <span class="nt">-S</span> <span class="nt">-n</span> <span class="nt">-u</span> <span class="nt">root</span> <span class="o">/</span><span class="nt">bin</span><span class="o">/</span><span class="nt">sh</span> <span class="nt">-c</span> <span class="s1">&#39;echo BECOME-SUCCESS-mtobhjchhszgqaaixzbsbsolwbuprmhn; LANG=de_DE.UTF-7 LC_ALL=de_DE.UTF-8 LC_MESSAGES=de_DE.UTF-8 /usr/bin/python /home/vagrant/.ansible/tmp/ansible-tmp-1469778394.25-104576843898387/apk; rm -rf \&quot;/home/vagrant/.ansible/tmp/ansible-tmp-1469778394.25-104576843898387/\&quot; &gt; /dev/null 2&gt;&amp;1&#39;</span> <span class="o">&amp;&amp;</span> <span class="nt">sleep</span> <span class="nt">0</span><span class="o">:</span> <span class="nt">not</span> <span class="nt">found</span><span class="err">\</span><span class="nt">r</span><span class="err">\</span><span class="nt">n</span> </pre></div> <p>Loggte ich mich ein und führe das Ansible Playbook manuell aus, kein Problem. Ich hatte xonsh als Login Shell eingerichtet und da läuft etwas schief. SSH ruft den Befehl per <code>exec</code> auf und da hat xonsh noch ein paar Probleme. Es gibt den Shell Befehl <code>exec</code> und die Python-Funktion <code>exec</code>. Diese sind schwierig zu unterscheiden und schon läuft es schief. Es wird an einem Fix gearbeitet. Der soll auch in den nächsten Tagen released werden. Bis dahin habe ich mich dazu entschieden meine Login-Shell wieder auf Bash umzustellen und daraus dann xonsh zu starten.</p> <p><del>Dazu benutze ich das File <code>~/.profile</code>. Dies ist dafür da Sachen beim Login auszuführen. Aber nur wenn es <code>~/.bash_profile</code> und <code>~/bash_login</code> nicht gibt. Ich erweitere also <code>~/.profile</code>.</del></p> <div class="highlight"><pre><span></span>[ -f /usr/local/bin/xonsh ] &amp;&amp; exec /usr/local/bin/xonsh </pre></div> <p><del>Wenn es <code>/usr/local/bin/xonsh</code> gibt dann führe es aus.</del></p> <p>Ich nahm an das es so funktionieren würde. Falsch gedacht. Es gab einige Probleme. Das schwerwiegendste war das LightDM mich nicht mehr einloggen wollte. Es liest beim einloggen die <code>~/.profile</code> und der <code>exec</code> Befehl behindert das ausführen von i3. Das starten von xonsh aus .profile heraus erschien mir als ein guter Weg. Aber sind wir mal ehrlich: Das ich xonsh nicht per <code>chsh</code> setzen kann, endet wohl oder übel in heftigstes, unsauberes gefrickel. Was macht man sonst an einem Samstag Vormittag? Nun kam ich zu folgender Lösung: Ich starte xonsh aus der <code>~/.bashrc</code> mit:</p> <div class="highlight"><pre><span></span>[ -f /usr/local/bin/xonsh ] &amp;&amp; exec /usr/local/bin/xonsh </pre></div> <p>Überraschung: die wird nicht immer geladen. SSH auf eine Alpine Linux Box warf mich in eine bash Shell. Also sollte man sich eine <code>~/.bash_profile</code> anlegen mit folgenden Inhalt:</p> <div class="highlight"><pre><span></span>if [ -f ~/.bashrc ]; then . ~/.bashrc fi </pre></div> <p>Nun geht erstmal wieder alles. Ein Workaround... aber was solls? Ich warte auf das nächste Krachen. Aber am meisten freue ich mich auf den Fix.</p>Meine neue Shell: xonsh2016-07-28T16:47:00+02:002016-07-28T16:47:00+02:00marvintag:xsteadfastx.org,2016-07-28:/2016/07/28/meine-neue-shell-xonsh/<p>Manchmal macht man Sachen die man selber nicht so wirklich versteht. Zum Beispiel hatte ich mich lange quer gestellt <a href="https://de.wikipedia.org/wiki/Zsh">ZSH</a> einzusetzen. Wieso eine alternative Shell? BASH ist doch sowas wie ein standard auf den Servern auf denen ich arbeite. Vor ein paar Jahren dann die erste Installation. Total abgeschreckt von …</p><p>Manchmal macht man Sachen die man selber nicht so wirklich versteht. Zum Beispiel hatte ich mich lange quer gestellt <a href="https://de.wikipedia.org/wiki/Zsh">ZSH</a> einzusetzen. Wieso eine alternative Shell? BASH ist doch sowas wie ein standard auf den Servern auf denen ich arbeite. Vor ein paar Jahren dann die erste Installation. Total abgeschreckt von der Konfiguration (bin bis Heute nicht dahinter gekommen wie das alles funktioniert), war mir <a href="">oh-my-zsh</a> ein Steigbügelhalter in die Welt von ZSH. Es ist eine Paket aus verschiedenen Plugins und Konfigurationen die das erste ZSH-Erlebnis, instant, beeindruckend gestaltet. Irgendwann war es mir zu langsam und ich stieg um auf <a href="https://github.com/sorin-ionescu/prezto">prezto</a>. Hat sich alles ein wenig flotter angefühlt und ich habe noch weniger verstanden was da im Hintergrund abläuft. Ja, man sollte sich die Sachen genauer anschauen und dann wäre es auch kein Problem. Aber manchmal fehlt mir Zeit weil ich meine Nase in tausend anderen Projekten habe. Es soll einfach funktionieren und the world a better place machen tun... oder zumindest meinen Alltag.</p> <p>Nun bin ich auf einen Beitrag auf der PyCon über <a href="http://xon.sh">xonsh</a> gestossen.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/uaje5I22kgE' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p>Es ist eine alternative Shell die in Python geschrieben ist und das Bindeglied zwischen Shell und Python sein möchte. Und das beste: Endlich eine Syntax in der Shell die ich mir merken kann. Nie wieder nachschauen wie <code>if</code> oder <code>for</code> in Bash funktionieren. "Nie wieder" ist ein harter Ausdruck. Selbst ich installiere xonsh nicht auf all meinen Servern. Es setzt Python 3.4 oder 3.5 voraus.</p> <p>Bevor ich ein paar Anwendungsbeispiele nenne, möchte ich auf mein Setup hinweisen. Auf <a href="https://github.com/xsteadfastx/batcave">GitHub</a> befinden sich meine Ansible Roles um meine Rechner einzurichten. Unter anderen auch <a href="https://github.com/xsteadfastx/batcave/tree/master/roles/xonsh">xonsh</a>.</p> <h2>Python und Shell Kommandos gemeinsam nutzen</h2> <p>Hier ein Beispiel für ein xonsh-Script das den Output von allen Docker-Commands nimmt und den Output in Files schreibt. Dies habe ich gebraucht um meine xonsh Extension <a href="https://github.com/xsteadfastx/xonsh-docker-tabcomplete">xonsh-docker-tabcomplete</a> zu testen. Ein Completer für Docker Kommandos die auch direkt auf die Docker-API zugreift. Danke <a href="https://github.com/docker/docker-py">docker-py</a> und Python als Shell.</p> <div class="highlight"><pre><span></span><span class="sd">&quot;&quot;&quot;Ein xonsh-Script um alle `--help` Outputs als Textfiles</span> <span class="sd">abzuspeichern.</span> <span class="sd">&quot;&quot;&quot;</span> <span class="kn">import</span> <span class="nn">os</span> <span class="kn">import</span> <span class="nn">re</span> <span class="c1"># `parser` ist ein Modul mit ein paar Regex Parsern.</span> <span class="kn">from</span> <span class="nn">docker_tabcomplete</span> <span class="kn">import</span> <span class="n">parser</span> <span class="c1"># Regex um Versionsnummer zu finden.</span> <span class="n">RE_VERSION</span> <span class="o">=</span> <span class="n">re</span><span class="o">.</span><span class="n">compile</span><span class="p">(</span><span class="sa">r</span><span class="s1">&#39;((?:\d+\.)?(?:\d+\.)?(?:\*|\d+))&#39;</span><span class="p">)</span> <span class="c1"># Hier nutze ich das erste mal ein wenig xonsh Magic.</span> <span class="c1"># Das `$()` macht genau das gleiche wie in Bash-Scripten. Es führt</span> <span class="c1"># Ein Kommando aus aus gibt den Output zurück. In diesem Fall den</span> <span class="c1"># Output von `docker --version` und sucht in Python per Regex nach</span> <span class="c1"># der Versionsnummer.</span> <span class="n">DOCKER_VERSION</span> <span class="o">=</span> <span class="n">re</span><span class="o">.</span><span class="n">search</span><span class="p">(</span><span class="n">RE_VERSION</span><span class="p">,</span> <span class="err">$</span><span class="p">(</span><span class="n">docker</span> <span class="o">--</span><span class="n">version</span><span class="p">))</span><span class="o">.</span><span class="n">group</span><span class="p">(</span><span class="mi">1</span><span class="p">)</span> <span class="c1"># Es wird ein Verzeichnis für die Docker Version angelegt falls es noch</span> <span class="c1"># nicht existiert.</span> <span class="k">if</span> <span class="ow">not</span> <span class="n">os</span><span class="o">.</span><span class="n">path</span><span class="o">.</span><span class="n">exists</span><span class="p">(</span><span class="s1">&#39;../tests/data/{}&#39;</span><span class="o">.</span><span class="n">format</span><span class="p">(</span><span class="n">DOCKER_VERSION</span><span class="p">)):</span> <span class="n">os</span><span class="o">.</span><span class="n">makedirs</span><span class="p">(</span><span class="s1">&#39;../tests/data/{}&#39;</span><span class="o">.</span><span class="n">format</span><span class="p">(</span><span class="n">DOCKER_VERSION</span><span class="p">))</span> <span class="c1"># `parser.commands` nimmt den Output von `docker --help` und parst</span> <span class="c1"># nach allen Docker Kommandos. Wieder muss nicht mit `subprocess`</span> <span class="c1"># gefummelt werden sondern wir nutzen einfach `$()` von xonsh um</span> <span class="c1"># an den Output zu gelangen.</span> <span class="n">COMMANDS</span> <span class="o">=</span> <span class="n">parser</span><span class="o">.</span><span class="n">commands</span><span class="p">(</span><span class="err">$</span><span class="p">(</span><span class="n">docker</span> <span class="o">--</span><span class="n">help</span><span class="p">))</span> <span class="c1"># Jetzt wird sogar Python mit Shell mit Python gemischt.</span> <span class="k">for</span> <span class="n">command</span> <span class="ow">in</span> <span class="n">COMMANDS</span><span class="p">:</span> <span class="k">with</span> <span class="nb">open</span><span class="p">(</span><span class="s1">&#39;../tests/data/{}/{}.stdout&#39;</span><span class="o">.</span><span class="n">format</span><span class="p">(</span><span class="n">DOCKER_VERSION</span><span class="p">,</span> <span class="n">command</span><span class="p">),</span> <span class="s1">&#39;w&#39;</span><span class="p">)</span> <span class="k">as</span> <span class="n">f</span><span class="p">:</span> <span class="c1"># Es wird die Python-Variabel `command`, die gerade in benutzung ist,</span> <span class="c1"># in das Shell-Kommando `docker ... --help` eingefügt. Magic.</span> <span class="n">f</span><span class="o">.</span><span class="n">write</span><span class="p">(</span><span class="err">$</span><span class="p">(</span><span class="n">docker</span> <span class="err">@</span><span class="p">(</span><span class="n">command</span><span class="p">)</span> <span class="o">--</span><span class="n">help</span><span class="p">))</span> </pre></div> <p>Dieses Script speichert man als <code>create_test_data.xsh</code> und kann es mit <code>xonsh create_test_data.xsh</code> ausführen.</p> <h2>Meine <code>~/.xonshrc</code></h2> <p>Yeah, endlich eine Shell Konfiguration in Python...</p> <div class="highlight"><pre><span></span><span class="kn">import</span> <span class="nn">os</span> <span class="kn">import</span> <span class="nn">shutil</span> <span class="kn">from</span> <span class="nn">xonsh.environ</span> <span class="kn">import</span> <span class="n">git_dirty_working_directory</span> <span class="c1"># XONSH ENVIRONMENT VARIABLES ############</span> <span class="c1"># Die Environment Variabeln werden wie Python Variabeln definiert.</span> <span class="c1"># Hier einige Beispiele. Diese sind vor allem um das Verhalten von</span> <span class="c1"># xonsh anzupassen.</span> <span class="c1"># xonsh supportet die bestehenden bash completion files.</span> <span class="c1"># Zumindestens die meisten. Ich hatte Probleme mit dem</span> <span class="c1"># Docker-Completion File und habe deswegen mein eigenes geschrieben.</span> <span class="err">$</span><span class="n">BASH_COMPLETIONS</span> <span class="o">=</span> <span class="p">[</span><span class="s1">&#39;/etc/bash_completion.d&#39;</span><span class="p">,</span> <span class="s1">&#39;/usr/share/bash-completion/completions&#39;</span><span class="p">]</span> <span class="err">$</span><span class="n">CASE_SENSITIVE_COMPLETIONS</span> <span class="o">=</span> <span class="bp">False</span> <span class="err">$</span><span class="n">SHELL_TYPE</span> <span class="o">=</span> <span class="s1">&#39;best&#39;</span> <span class="err">$</span><span class="n">SUPPRESS_BRANCH_TIMEOUT_MESSAGE</span> <span class="o">=</span> <span class="bp">True</span> <span class="err">$</span><span class="n">VC_BRANCH_TIMEOUT</span> <span class="o">=</span> <span class="mi">5</span> <span class="err">$</span><span class="n">XONSH_COLOR_STYLE</span> <span class="o">=</span> <span class="s1">&#39;monokai&#39;</span> <span class="c1"># OTHER ENVIRONMENT VARIABLES ############</span> <span class="err">$</span><span class="n">EDITOR</span> <span class="o">=</span> <span class="s1">&#39;vim&#39;</span> <span class="err">$</span><span class="n">PYTHONIOENCODING</span> <span class="o">=</span> <span class="s1">&#39;UTF-8&#39;</span> <span class="err">$</span><span class="n">LC_ALL</span> <span class="o">=</span> <span class="s1">&#39;C.UTF-8&#39;</span> <span class="err">$</span><span class="n">SSL_CERT_FILE</span> <span class="o">=</span> <span class="s1">&#39;/etc/ssl/certs/ca-certificates.crt&#39;</span> <span class="c1"># ALIASES ################################</span> <span class="c1"># Aliase sind in einem dict gespeichert das einfach erweitert werden kann.</span> <span class="n">aliases</span><span class="p">[</span><span class="s1">&#39;exec&#39;</span><span class="p">]</span> <span class="o">=</span> <span class="n">aliases</span><span class="p">[</span><span class="s1">&#39;xexec&#39;</span><span class="p">]</span> <span class="n">aliases</span><span class="p">[</span><span class="s1">&#39;ll&#39;</span><span class="p">]</span> <span class="o">=</span> <span class="s1">&#39;ls -la&#39;</span> <span class="n">aliases</span><span class="p">[</span><span class="s1">&#39;tmux&#39;</span><span class="p">]</span> <span class="o">=</span> <span class="s1">&#39;tmux -2&#39;</span> <span class="c1"># Jetzt wird es spannend: Wir können Funktionen oder Lambdas als</span> <span class="c1"># Aliase definieren. Hier der Fall das Mosh eine bestimmte</span> <span class="c1"># Environment Variabel gesetzt haben muss.</span> <span class="k">def</span> <span class="nf">_alias_mosh</span><span class="p">(</span><span class="n">args</span><span class="p">,</span> <span class="n">stdin</span><span class="o">=</span><span class="bp">None</span><span class="p">):</span> <span class="sd">&quot;&quot;&quot;A function to use as alias to get mosh running.</span> <span class="sd"> There is a strange problem with mosh and xonsh. I have to set $SHELL to</span> <span class="sd"> /bin/bash before running it. It should work with this little hack.</span> <span class="sd"> &quot;&quot;&quot;</span> <span class="n">os</span><span class="o">.</span><span class="n">environ</span><span class="p">[</span><span class="s1">&#39;SHELL&#39;</span><span class="p">]</span> <span class="o">=</span> <span class="s1">&#39;/bin/bash&#39;</span> <span class="n">args</span><span class="o">.</span><span class="n">insert</span><span class="p">(</span><span class="mi">0</span><span class="p">,</span> <span class="s1">&#39;mosh&#39;</span><span class="p">)</span> <span class="n">cmd</span> <span class="o">=</span> <span class="s1">&#39; &#39;</span><span class="o">.</span><span class="n">join</span><span class="p">(</span><span class="n">args</span><span class="p">)</span> <span class="n">os</span><span class="o">.</span><span class="n">system</span><span class="p">(</span><span class="n">cmd</span><span class="p">)</span> <span class="n">aliases</span><span class="p">[</span><span class="s1">&#39;mosh&#39;</span><span class="p">]</span> <span class="o">=</span> <span class="n">_alias_mosh</span> <span class="c1"># GIT ####################################</span> <span class="err">$</span><span class="n">FORMATTER_DICT</span><span class="p">[</span><span class="s1">&#39;branch_color&#39;</span><span class="p">]</span> <span class="o">=</span> <span class="k">lambda</span><span class="p">:</span> <span class="p">(</span><span class="s1">&#39;{BOLD_INTENSE_RED}&#39;</span> <span class="k">if</span> <span class="n">git_dirty_working_directory</span><span class="p">(</span><span class="n">include_untracked</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span> <span class="k">else</span> <span class="s1">&#39;{BOLD_INTENSE_GREEN}&#39;</span><span class="p">)</span> <span class="c1"># PATH ###################################</span> <span class="c1"># Die Path Variabel ist eine Liste die ich einfach extende.</span> <span class="err">$</span><span class="n">PATH</span><span class="o">.</span><span class="n">extend</span><span class="p">(</span> <span class="p">[</span> <span class="err">$</span><span class="n">HOME</span> <span class="o">+</span> <span class="s1">&#39;/.local/bin&#39;</span><span class="p">,</span> <span class="err">$</span><span class="n">HOME</span> <span class="o">+</span> <span class="s1">&#39;/miniconda3/bin&#39;</span><span class="p">,</span> <span class="err">$</span><span class="n">HOME</span> <span class="o">+</span> <span class="s1">&#39;/node_modules_global/bin&#39;</span> <span class="p">]</span> <span class="p">)</span> <span class="c1"># XONTRIB ################################</span> <span class="c1"># Hier habe ich ein paar Erweiterungen enabled. Bei manchen schaue ich</span> <span class="c1"># ob der Befehl auf denen sie beruhen überhaupt da und einsatzfähig ist</span> <span class="c1"># bevor ich sie anschalte. Was nützt mir `apt-get` wenn ich nicht auf</span> <span class="c1"># einem Debian/Ubuntu System bin oder Docker nicht installiert ist?</span> <span class="c1"># Dies teste ich mit `shutil.which`.</span> <span class="n">xontrib</span> <span class="n">autoxsh</span> <span class="n">vox_tabcomplete</span> <span class="k">if</span> <span class="n">shutil</span><span class="o">.</span><span class="n">which</span><span class="p">(</span><span class="s1">&#39;apt-get&#39;</span><span class="p">):</span> <span class="n">xontrib</span> <span class="n">apt_tabcomplete</span> <span class="k">if</span> <span class="n">shutil</span><span class="o">.</span><span class="n">which</span><span class="p">(</span><span class="s1">&#39;docker&#39;</span><span class="p">):</span> <span class="n">xontrib</span> <span class="n">docker_tabcomplete</span> </pre></div> <h2>Ein wenig Rescue</h2> <p>Natürlich kann alles noch ein wenig buggy sein. Einmal gab es Probleme mit den Schreibrechten auf dem History-File und ich konnte kein Terminal mehr öffnen oder mich einloggen. Was da helfen kann: Per SSH und anderer Shell draufschalten und die Shell ändern.</p> <div class="highlight"><pre><span></span>ssh meinuser@meinrechner sh </pre></div> <h2>Fazit</h2> <p>Das ist natürlich nur ein Bruchteil der Möglichkeiten und Sachen die man ausprobieren kann. Es ist ein Anfang und bis jetzt bin ich begeistert. Also einfach mal das Youtube Video anschauen und ausprobieren. Muss ja nicht gleich die Standard-Shell werden wie bei mir ;-).</p>Touché Amoré - Palm Dreams2016-07-26T16:28:00+02:002016-07-26T16:28:00+02:00marvintag:xsteadfastx.org,2016-07-26:/2016/07/26/touche-amore-palm-dreams/<p>Touché Amoré bringen am 19.09. ihr neues Album raus. Nun ist schon das erste Video online. Eine Hommage an einen meiner Lieblingsfilme "Mulholland Drive".</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/aCj2VhmnluE' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p><p>Touché Amoré bringen am 19.09. ihr neues Album raus. Nun ist schon das erste Video online. Eine Hommage an einen meiner Lieblingsfilme "Mulholland Drive".</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/aCj2VhmnluE' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>GERMANY GERMANY in einem Wald2016-07-26T15:09:00+02:002016-07-26T15:09:00+02:00marvintag:xsteadfastx.org,2016-07-26:/2016/07/26/germany-germany-in-einem-wald/<p>Es ist schon fast zuviel Sommer. Diese wenigen Tage an denen es warm ist sind dann so warm das es einem zuviel ist. Musik kann da einem helfen alles ein wenig besser zu ertragen. Musik wie die von GERMANY GERMANY, die für dieses Video ihr Equipment in einen Wald gebracht …</p><p>Es ist schon fast zuviel Sommer. Diese wenigen Tage an denen es warm ist sind dann so warm das es einem zuviel ist. Musik kann da einem helfen alles ein wenig besser zu ertragen. Musik wie die von GERMANY GERMANY, die für dieses Video ihr Equipment in einen Wald gebracht haben.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/v1vq-F0aEj8' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Making Of Wolfsburg Unlimited2016-07-22T09:43:00+02:002016-07-22T09:43:00+02:00marvintag:xsteadfastx.org,2016-07-22:/2016/07/22/making-of-wolfsburg-unlimited/<p>Das Kunstmuseum Wolfsburg hat ein kleines Zeitraffer-Making-Of zu Wolfsburg Unlimited online gestellt.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/YxYU9xVJcJE' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p><p>Das Kunstmuseum Wolfsburg hat ein kleines Zeitraffer-Making-Of zu Wolfsburg Unlimited online gestellt.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/YxYU9xVJcJE' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Floating Piers Webcam2016-06-27T14:19:00+02:002016-06-27T14:19:00+02:00marvintag:xsteadfastx.org,2016-06-27:/2016/06/27/floating-piers-webcam/<p>Christo hat es mal wieder getan. Sein neues Kunstwerk <a href="http://www.thefloatingpiers.com/">The Floating Piers</a> ist zum Pilgerort geworden. Zehntausende Besucher tummeln sich auf dem See <a href="https://de.wikipedia.org/wiki/Iseosee">Iseosee</a> in Italien. So viele Menschen das sie es teilweise sogar schließen mussten. Ich schaffe es leider nicht das Werk zu besuchen. Ein kleiner Trost ist diese …</p><p>Christo hat es mal wieder getan. Sein neues Kunstwerk <a href="http://www.thefloatingpiers.com/">The Floating Piers</a> ist zum Pilgerort geworden. Zehntausende Besucher tummeln sich auf dem See <a href="https://de.wikipedia.org/wiki/Iseosee">Iseosee</a> in Italien. So viele Menschen das sie es teilweise sogar schließen mussten. Ich schaffe es leider nicht das Werk zu besuchen. Ein kleiner Trost ist diese Webcam...</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/CdBKNCPKBE8' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>A Conversation with Koko2016-06-15T13:21:00+02:002016-06-15T13:21:00+02:00marvintag:xsteadfastx.org,2016-06-15:/2016/06/15/a-conversation-with-koko/<p>In einer Zeitschrift fanden wir einen Artikel über den Gorilla <a href="https://de.wikipedia.org/wiki/Koko_(Gorilla)">Koko</a> und sein Kätzchen <a href="http://www.koko.org/koko's-kittens">ALL BALL</a>. Koko wurde die amerikanische Zeichensprache beigebracht und dies schafft Möglichkeiten mit Koko zu kommunizieren, sich über ihre Gedankenwelt auszutauschen. Eines Tages teilte Koko mit, dass sie eine Katze zum spielen haben möchte. Nach diesem …</p><p>In einer Zeitschrift fanden wir einen Artikel über den Gorilla <a href="https://de.wikipedia.org/wiki/Koko_(Gorilla)">Koko</a> und sein Kätzchen <a href="http://www.koko.org/koko's-kittens">ALL BALL</a>. Koko wurde die amerikanische Zeichensprache beigebracht und dies schafft Möglichkeiten mit Koko zu kommunizieren, sich über ihre Gedankenwelt auszutauschen. Eines Tages teilte Koko mit, dass sie eine Katze zum spielen haben möchte. Nach diesem kurzen Artikel googelte ich herum und fand folgende Dokumentation. Ich hatte die ganze Zeit feuchte Augen und meine Verzweifelung gegenüber vielen Menschen und ihrem Anspruch an die "Krönung der Schöpfung" wurde größer und größer. Was würde es für die Welt bedeuten wenn wir nicht nur auf uns sondern auch auf die anderen Geschöpfe achten würden?</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/8oh1uhrdc6w' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Mat Marrash und seine Großformat Fotografie2016-06-07T12:47:00+02:002016-06-07T12:47:00+02:00marvintag:xsteadfastx.org,2016-06-07:/2016/06/07/mat-marrash-und-seine-grossformat-fotografie/<p>Kleine Dokumentation über den Fotografen <a href="http://matmarrash.com/">Mat Marrash</a> über Großformat Fotografie. Und ich verstehe ihn total. Auch wenn ich keine Großformat-Kamera besitze. Meine digitale Kamera habe ich seit fast 2 Jahren nicht mehr benutzt.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/f0_Nax7PBtM' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p><p>Kleine Dokumentation über den Fotografen <a href="http://matmarrash.com/">Mat Marrash</a> über Großformat Fotografie. Und ich verstehe ihn total. Auch wenn ich keine Großformat-Kamera besitze. Meine digitale Kamera habe ich seit fast 2 Jahren nicht mehr benutzt.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/f0_Nax7PBtM' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Richtig spitzen mit David Rees2016-05-27T11:26:00+02:002016-05-27T11:26:00+02:00marvintag:xsteadfastx.org,2016-05-27:/2016/05/27/richtig-spitzen-mit-david-rees/<p><img alt="Book Cover of How to sharpen pencils" src="http://xsteadfastx.org/images/how-to-sharpen-pencils.jpg"></p> <p><a href="https://en.wikipedia.org/wiki/David_Rees_(cartoonist)">David Rees</a> Namen habe ich das erste mal gelesen als ich nach Arten des Anspitzens gesucht habe. Er hat nämlich ein ganzes <a href="http://www.amazon.com/How-Sharpen-Pencils-Theoretical-Contractors/dp/1612193269">Buch</a> darüber geschrieben und betreibt ein <a href="http://www.artisanalpencilsharpening.com/">Artisanal Pencil Sharpening</a> Dienstleister. Nach dem ersten Video wird man sich unsicherer ob es sich dabei um eine ganz klare Satire handelt …</p><p><img alt="Book Cover of How to sharpen pencils" src="http://xsteadfastx.org/images/how-to-sharpen-pencils.jpg"></p> <p><a href="https://en.wikipedia.org/wiki/David_Rees_(cartoonist)">David Rees</a> Namen habe ich das erste mal gelesen als ich nach Arten des Anspitzens gesucht habe. Er hat nämlich ein ganzes <a href="http://www.amazon.com/How-Sharpen-Pencils-Theoretical-Contractors/dp/1612193269">Buch</a> darüber geschrieben und betreibt ein <a href="http://www.artisanalpencilsharpening.com/">Artisanal Pencil Sharpening</a> Dienstleister. Nach dem ersten Video wird man sich unsicherer ob es sich dabei um eine ganz klare Satire handelt. Die Wahrheit liegt wohl irgendwo dazwischen. Man spricht hierbei von einem Buch was gleichzeitig eine Parodie ist, aber auch die technische Referenz wenn es um das Anspitzen von Bleistiften geht. David Rees scheint sich in Themen des Alltags regelrecht reinzugraben um die richtigen Wege und Methoden zu finden. Bleistifte sind das eine... er hat aber auch eine Sendung auf National Geographic mit dem Namen <a href="http://channel.nationalgeographic.com/going-deep-with-david-rees/">Going deep with David Rees</a>. Dort sind die Themen etwas "wie schüttel ich Hände", "wie stelle ich Eiswürfel her" oder auch "wie grabe ich ein Loch". Ich habe nun schon ein paar Folgen gesehen und fühle mich wahnsinnig amüsiert.</p> <p><a href="http://www.newyorker.com/culture/sarah-larson/david-rees-going-deep-going-ridiculous">Hier</a> gibt es einen tollen Artikel über ihn und er war auch mal zu Gast im <a href="http://www.erasable.us/episode/15">Erasable Podcast</a> mit dem passenden Episodentitel "The Joke is There is No Joke (with David Rees)". Ich bin Fan.</p> <p><span class="videobox"> <iframe src="//player.vimeo.com/video/60718161?title=0&amp;byline=0&amp;portrait=0" width="640" height="390" frameborder="0" webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/KabOfnbS4TQ' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/b_ZlrUx6MtM' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/98otYZo6TYA' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Wolfsburg Unlimited2016-05-24T09:26:00+02:002016-05-24T09:26:00+02:00marvintag:xsteadfastx.org,2016-05-24:/2016/05/24/wolfsburg-unlimited/<p><em>Wolfsburg Unlimited - Eine Stadt als Weltlabor: 24.04.2016 - 11.09.2016 Kunstmuseum Wolfsburg</em></p> <p>Da standen wir vor dem architektonischen Plänen eine Wasserstraße durch die Wolfsburger Innenstadt zu fräsen. Wasser gegen Beton und Granit. Ich mag die Idee. Es gehört zur Grundaufgabe im ersten Semester an der Kunstakademie Nürnberg aus …</p><p><em>Wolfsburg Unlimited - Eine Stadt als Weltlabor: 24.04.2016 - 11.09.2016 Kunstmuseum Wolfsburg</em></p> <p>Da standen wir vor dem architektonischen Plänen eine Wasserstraße durch die Wolfsburger Innenstadt zu fräsen. Wasser gegen Beton und Granit. Ich mag die Idee. Es gehört zur Grundaufgabe im ersten Semester an der Kunstakademie Nürnberg aus dem Plärrer, die Drehscheibe des öffentlichen Nahverkehrs, in etwas annehmbares zu verwandeln. Etwas das weniger nach grauen Leben aussieht und mehr nach dem wie sich Nürnberg anfühlt wenn man durch die Altstadt spaziert, die sich nur wenige Meter weit weg befindet. Eine Idee war es den Platz einfach zu fluten. Etwas mehr Venedig und weniger graue Wüste. Den da wo Wasser ist, kann etwas nicht völlig entstellt sein. Jahre später stehen wir vor dem Plan des Kunstmuseums das gleiche mit der kompletten Innenstadt zu tun. Straßen aus Wasser. Auf den Darstellungen spielen Kinder am Ufer. Sie scheinen in einer schöneren Stadt groß zu werden als ich es damals tat. Auch wenn alles nur eine Illusion ist. Vor meiner Geburt bestand die komplette Fußgängerzone aus befahrbaren Straßen. Was ein Paradox. Die Menschen an den Rand geschoben und das Auto im Mittelpunkt. Die Prioritäten dieser Stadt klar verteilt. Ich kenne Wolfsburg nicht anders. Nur das seit meiner Geburt die Fußgängerzone exklusiv begehbar war. Eine kleine Metamorphose, erst Auto, dann Mensch und nun Wasser. Am schönsten wenn dieses Element uns den Platz raubt. Ein Hauch Venedig, in nur nicht so schön. Aber ein Versuch wert.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/22986111429/"><img src="https://farm1.staticflickr.com/707/22986111429_967b88579e_b.jpg" alt="Wolfsburg Tristesse"></a></p> <p>Dieser Plan zur Umgestaltung ist Teil der Ausstellung <a href="http://www.kunstmuseum-wolfsburg.de/ausstellungen/wolfsburg-unlimited-eine-stadt-als-weltlabor/">Wolfsburg Unlimited</a>. Wir Wolfsburger wenden viel Zeit auf uns von diesem Ort zu distanzieren. Sie zu leugnen und ihre Bedeutung für uns, von uns, auf ein Minimum zu mindern. Es ist die Stadt in der ich geboren und aufgewachsen bin. Zwischendurch war ich weg und bin nun wieder zurück.</p> <p>Wolfsburg sehnt sich nach Identifikation, weit weg der Selbstverleugnung. Als wir vor einigen Monaten in Paris waren, sprach uns ein in Frankfurt lebender Amerikaner an und fragte woher wir kommen. Als wir Wolfsburg erwähnten sagte er: "Sehr häßliche Stadt, aber gute Kunst". Es spricht nichts dagegen eine Arbeitermetropole zu sein und Kunst ist immer ein guter Weg aus der Identifikationskrise. Ich ziehe oft den Vergleich mit Manchester. Ja, weit hergeholt, aber aus dieser grauen Stadt sind die vielleicht besten und einflußreichsten Bands hervorgegangen. Und das mit Musik die komplett diese Farben der Stadt in ihrem Sound vor sich her trugen. Was mich am meisten an der Volkswagen Krise bewegt ist die Beflecktheit auf unseren einzigen Identifikationsstifter. Stirbt der Konzern, stirbt Wolfsburg mit ihm und ihrem letzten Hauch Bedeutung. Nicht das uns der jemals zugesprochen wurde. Trotzdem blieb uns etwas auf das wir ein Stolz ähnliches Gefühl abbilden konnten. Oder es uns zumindestens einbilden konnten.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/22726907983/"><img src="https://farm6.staticflickr.com/5649/22726907983_640799cd1f_b.jpg" alt="Wolfsburg Tristesse"></a></p> <p>Vielleicht ist diese Ausstellung genau zur richtigen Zeit. Ein letzter Ankerwurf in dieser schwierigen Zeit. Ich hatte mir nie Gedanken über Heimat gemacht. Es gab nur irgendwo Wolfsburg und ich zog nie große Vergleiche zu der handvoll anderer Städte die ich kannte. Dann kam ich an meinen Ausbildungsplatz und auf einmal musste ich mich täglich für den Ort rechtfertigen an dem ich geboren bin. Wolfsburg erzeugt bestimmt nicht diese romantischen Fantasien wie Berlin oder Paris. Wir leben hier nur.</p> <p>Im ersten Raum der Ausstellung wurden alte archäologische Artefakte aus dem Stadtgebiet aufgebahrt. Überreste eines alten Kanus aus dem städtischen Badesee und alte Gemälde aus der namensgebenden Burg. Der See ist Teil eines streng durchgeplanten Naherholgungsgebietes im Norden der Stadt. Eine Fläche die den Arbeitern Ablenkung schenken soll von den, im Akkord vorbei rauschenden, Autoteilen am Band in der Fabrik. Hier steht auch die Volkswagen Arena. Seit dem wir in der ersten Bundesliga spielen bieten wir noch eine, zum Wochenende potenzierende, Angriffsfläche für den Spott der Anderen. Keine Tradition und keine Fans. Genau das sind wir. Vielleicht schon einmal was von uns gehört. Schrödingers Fans. Gerne könnte ich darauf verzichten. Im nächsten Raum treffen wir auf den Erdenker dieser Stadt: Adolf Hitler. Er gründete die Stadt des KdF-Wagens. Ein großer Brocken mit dem man leben muss. Anscheinend suchen wir uns gerne Städte aus in denen das 3. Reich eine große Rolle gespielt hat. Unterbewusst. Unser geliebtes Nürnberg und nun Wolfsburg. Vorbei zogen wir an Fotos der hakenkreuzbeflakten Grundsteinlegung und Hitlers erster Käfer-Skizze. Danach, chronologisch angeordnet, Wirtschaftswunder, die deutsche Grundspießigkeit und der Beginn meiner eigenen Zeitrechnung. Meine zwei Bewerbungen, gescheitert, und das letzte 3/4 Jahr voller Schuld und Ungewissheiten. Überall in der Stadt in Herzen eingefasste Volkswagen Logos, Je suis Volkswagen, Segnungen für VW in den gläubigen Gemeinden. Uns ist klar, die Wurzeln dieser Stadt sind die nationalen und internationalen Autokäufer. Meine eigenen Wurzeln liegen in den Generationen von Arbeitern meiner Familie drüben auf der anderen Seite des Mittellandkanals.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/23354048975/"><img src="https://farm6.staticflickr.com/5778/23354048975_dd7e093f88_b.jpg" alt="Wolfsburg Tristesse"></a></p> <p>Die Stadt versucht viel um das Andenken an diese Zeit zu bewahren. Auch wenn wir es nicht so mit Tradition haben, trotzdem stehen wir in diesem riesigen Schatten. So groß das man ihm kaum entfliehen kann. Ich habe mal eine Geschichte gehört, dass das "Wolf" im Namen der Stadt dem Spitznamen des Führers entsprungen ist. Aus dieser Geschichte entstammt der Name der Kirchengemeinde "Lammburg". Immer präsent. Wir schreiten weiter die Wand der VW Geschichte ab und landen an einer Fotografie von Douglas Gordon. Er steht mit einem Schild mit der Aufschrift "Psycho" an Ende der Braunschweiger Straße. Wie ein Anhalter der zu seiner eigenen Ausstellung mitgenommen werden will. Am Ende der U-förmig angeordneten Zeitleiste, im zweiten Raum, ein Ad-Busting Plakat das auf Dieselgate eingeht. Dieses hing zur großen Klimakonfernz in Paris. "Yes we cheated". Deneben ein Monitor der den Versuch eines nicht gut aussehenden, aufgedunsenen, fiebrig wirkenden Martin Winterkorn zeigt, sich bei der Welt für die Folgen des Kapitalismus zu entschuldigen. Zu diesem Zeitpunkt war schon alles gelaufen. Ertappt. Den meisten Arbeitern war der Wettstreit über die verkauften Autos mit Toyoto relativ egal. Trotzdem überwand man die legalen Mittel um noch mehr zu verkaufen und schuf damit den Abgrund in den ich jeden Tag auf dem Weg zur Arbeit blicken muss.</p> <p>Wir verschwanden schnell hinter die nächste Tür zum Hauptwerk der Ausstellung. Wir verliessen die "Hall of Fame" und betraten "Midwest". Ein amerikanischer Verladeort, aufgefüllt mit Containern, inklusive Bodenbelag, Müll und Schlafplätzen für Obdachlose. Flackernes Industrielicht aus den Straßenlaternen und das Zirpen von Grillen. Betritt man dann das Innere dieser Containerstadt offenbart sich ein Autokino inklusive Snackbar. Gezeigt wird eine filmische Choreografie des Künstlers Julian Rosefeldt. So länger man dort umher wandert verliert man das Empfinden und fühlt sich als sei man durch die Schranktür in ein häßliches Narnia getreten. Man befindet sich nicht mehr im Inneren eines sonst so sauberen Whitecubes. Der feine Museumsboden zum Teil überbetoniert und mit Schotter überseht, so ist es kaum möglich mehr zu erahnen wo man sich gerade befindet. Die Container bilden eine Schlucht und türmen sich bis unters Museumsdach. Hinter der Autokinoleinwand ein falscher Sonnenuntergang. Weit in der Ferne, die Sonne fast nicht mehr sichtbar. Das Kino wirkt durch Spiegel so breit, das man nicht mehr an einen abgeschlossenen Raum denken kann. Selbst der Blick von der Empore macht die Szenerie surreal.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/23354048325/"><img src="https://farm6.staticflickr.com/5794/23354048325_3fd6712e1e_b.jpg" alt="Wolfsburg Tristesse"></a></p> <p>Der nächste Raumabschnitt im Stadtlabor heißt "Gallerie Nordhoff". Benannt nach dem wichtigen Vorstandsvorsitzenden in den 60er Jahren. In der wichtigsten Zeit Wolfsburgs. Die Zeit die den größten Wohlstand brachte und die Stadtlandschaft prägte mit ihren kleinen, spießigen Ein-Familien Häusern. Platz für 1 1/2 Kindern, Frau und Hund. Er war vielleicht wichtiger als jeder Bürgermeister der Stadt. Am Anfang diesen Ausstellugnsbereich wird ein kurzer Film aus der Zeit gezeigt. Er könnte den Familienalben meiner Großeltern entsprungen sein. Allesamt Arbeiter dieser Stadt. Kinderfotos meines Vaters vor den Hochhausanlagen. Dort liegen die Grundwerte und Familiengeschichten die diese Stadt ausmachen. Etwas an diesen alten Filmaufnahmen holt etwas tief vergrabenes in mir hervor. Eine mit Familie verknüpfte Melancholie. Etwas was mich meine Eltern in die Arme schließen lassen will. Noch mehr als sonst.</p> <p>Ein Bild finde ich besonders beeindruckend. Heinrich Nordhoff steht erhoben auf einer Empore, nah in die Kamera blickend, hinter sich die versammelte Belegschaft des Wolfsburger Werkes. Aufgereiht, die gesammte Straße entlang.</p> <p>In den nächsten Räumen beginnt die zeitgenössische Auseinandersetzung mit Wolfsburg. Ich überfliege viele Arbeiten. Ich schnappe einige Bilder und Textfetzen auf. Letzten Sommer, kurz nachdem ich wieder nach Wolfsburg gezogen bin, begann der Abgassskandal. Zu meinen allgemeinen Zweifeln kam auch noch dieses über dem Kopf hängende Schwert mit der Aufschrift "Dieselgate". Meine Dunkelkammer noch nicht aufgebaut, wollte ich meine ersten Bilder in der Innenstadt machen. Ich tue mir mit so einer Aufgabe schwer, aber es sollte eine Auseinandersetzung mit meinem gestörten Verhältnis mit meiner Heimat sein. In den Arbeiten der hier ausgestellten Künstlern fehlt mir der Bezug. Dies liegt vielleicht daran, das der Museumsleiter Wolfsburg eher als eine Extraktion des gesamten Landes sieht. Alles konnte ich an diesem Mittag nicht verarbeiten. Es sind zu viele Eindrücke die mich beschäftigen. Eindrücke die alle Stufen einer Emotionstabelle abdecken könnten. Oft gehe ich in Ausstellungen und bin danach über alle Ohren inspiriert oder auch bewegt. Diese Ausstellung mit ihrer Auflistung der Facetten und Geschichten der Stadt trifft mich ins Herz. Und das ziemlich hart und unvorbereitet. Und am Ende weiß ich immer noch nicht wie es in dieser Stadt weitergehen soll.</p>Melancholia: Depression on Film2016-05-23T10:14:00+02:002016-05-23T10:14:00+02:00marvintag:xsteadfastx.org,2016-05-23:/2016/05/23/melancholia-depression-on-film/<p><img alt="Ophelia von John Everett Millais" src="http://xsteadfastx.org/images/ophelia.jpg"></p> <p>Melancholia ist einer dieser Lieblingsfilme die ich nicht ein zweites mal schauen konnte. So intensiv habe ich ihn in Erinnerung. Der <a href="https://www.youtube.com/channel/UCJkMlOu7faDgqh4PfzbpLdg">Nerdwriter</a> hat über die Darstellung von Depression in dem Film ein Video gemacht.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/FPkANZ9HGWE' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p><p><img alt="Ophelia von John Everett Millais" src="http://xsteadfastx.org/images/ophelia.jpg"></p> <p>Melancholia ist einer dieser Lieblingsfilme die ich nicht ein zweites mal schauen konnte. So intensiv habe ich ihn in Erinnerung. Der <a href="https://www.youtube.com/channel/UCJkMlOu7faDgqh4PfzbpLdg">Nerdwriter</a> hat über die Darstellung von Depression in dem Film ein Video gemacht.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/FPkANZ9HGWE' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Das Thema Anspitzer ist durch oder auch KUM Masterpiece2016-05-17T13:50:00+02:002016-05-17T13:50:00+02:00marvintag:xsteadfastx.org,2016-05-17:/2016/05/17/das-thema-anspitzer-ist-durch-oder-auch-kum-masterpiece/<p><img alt="KUM Masterpiece" src="http://xsteadfastx.org/images/kummasterpiece.jpg"></p> <p>Auf meinem Schreibtisch stehen gerade 6 Anspitzer. Es wären 7 wenn ich den einen nicht wirklich immer im Rucksack bei mir tragen würde. Als ich anfing mehr mit dem Bleistift zu schreiben hätte ich niemals gedacht das es ein großes Probleme wäre einen vernünftigen und portablen Spitzer zu finden. Ich …</p><p><img alt="KUM Masterpiece" src="http://xsteadfastx.org/images/kummasterpiece.jpg"></p> <p>Auf meinem Schreibtisch stehen gerade 6 Anspitzer. Es wären 7 wenn ich den einen nicht wirklich immer im Rucksack bei mir tragen würde. Als ich anfing mehr mit dem Bleistift zu schreiben hätte ich niemals gedacht das es ein großes Probleme wäre einen vernünftigen und portablen Spitzer zu finden. Ich bin naiv zum Staples gefahren und habe einen Spitzer in einer Plastikdose gekauft, mit der Hoffnung diese würde dicht halten und gut spitzen. Nach 5 Tagen hatte ich dann keine Späne mehr im Behälter, sondern nur noch im Rucksack. Dann hatte ich diverse andere Spitzer die zwar den Span sicher aufbewahrt haben, die Spitze aber mehr zum Zeichnen geeignet war als zum schreiben.</p> <p>Erstmal musste ich mir klar werden was für eine Spitze ich haben will. Am besten gefällt mir eine lange und sehr spitze Bleistift spitze. Ist sich zwar spitz aber der Winkel ist weiter, erreicht man nach kurzem schreiben gleich einen breiteren Bereich und die Striche werden dicker. Ich mag eher dünne Striche und mit einer längeren Spitze habe ich auch mehr davon. Sonst wird es zu lästig immer und immer wieder nach zuspitzen. Zuhause habe ich immer den <a href="http://www.amazon.de/Dahle-00155-20094-Bleistiftspitzmaschine-Stiftdurchmesser-grau/dp/B001IVHRRY">Dahl 155</a> benutzt. Der spitzt eine sehr sehr schöne Spitze. Und unter uns: Spitzen hat etwas sehr meditatives. Das systematische Abtragen von Altlasten, in gleichmäßigen Zügen, bis das Endergebnis, untermalt von dem wunderbaren holzigen Geruch, die erneuerte Freude am Schreiben hervorbringt.</p> <p>Nun las ich in einem <a href="http://www.lexikaliker.de/2015/03/kum-masterpiece/">Blogartikel</a> über den KUM Masterpiece. Ein Handspitzer der eine wahnsinnige Spitze zaubert. Dies erreicht er durch zwei Phasen, die durch zwei verschiedene Öffnungen im Spitzer durcharbeitet werden. In der ersten Phase entledigt er dem Stift einiges an Holz bis er genug Grafit freigelegt hat. Der Anspitzer verfügt über einen Auto-Stopp. Im zweiten Loch spitzt er nur das Grafit zu einer ordentlichen Spitze. Leider hobelt er auch noch ein wenig Holz ab, so bleibt eine kleine Kerbe im Stift. Meine Erfahrungen mit dem <a href="http://www.staedtler.de/de/produkte/bleistifte-zubehoer/bleistifte/mars-lumograph-100-bleistift-in-bester-qualitaet/">Staedtler Mars Lumograph</a> haben mich so begeistert das ich alle in meinem Umfeld noch mehr mit meiner Faszination über Bleistifte nerve als sonst schon.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/2Xk8Ax1rQcA' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p>Den KUM Masterpiece gibt es nicht bei Amazon oder anderen Händlern, man muss ihn direkt bei KUM <a href="http://www.kum.net/index.php/masterpiece/">bestellen</a>. Ich habe es nicht bereut.</p>WSUS Offline und Ansible2016-05-13T12:52:00+02:002016-05-13T12:52:00+02:00marvintag:xsteadfastx.org,2016-05-13:/2016/05/13/wsus-offline-und-ansible/<p>Ansible ist schon sehr nice. Benutzt man es mit Windows, ist es auch noch nice aber manchmal fühlt man sich dabei wie ein Ritt auf einem Vulkan, durch die Hölle... oder sowas. Meist funktioniert es aber was im Hintergrund passiert gleicht Feen-Magie.</p> <p>Da wir es wagen einen Proxy zu benutzen …</p><p>Ansible ist schon sehr nice. Benutzt man es mit Windows, ist es auch noch nice aber manchmal fühlt man sich dabei wie ein Ritt auf einem Vulkan, durch die Hölle... oder sowas. Meist funktioniert es aber was im Hintergrund passiert gleicht Feen-Magie.</p> <p>Da wir es wagen einen Proxy zu benutzen scheint es völlig unmöglich zu sein an Windows Updates zu kommen. Und in letzter Zeit scheinen auch meine Workarounds nicht zu fruchten. Immer wieder hängen die Updates tagelang in der Pipeline und können nicht runtergeladen oder installiert werden. Also schaute ich mir mal wieder einen alten Bekannten an: <a href="http://www.wsusoffline.net/">WSUS Offline Update</a>. Ein Paket aus Scripten das Updates zentral runterlädt und dann offline installiert werden kann. Das schöne daran: Das Download-Script liegt auch in Shell vor und kann so direkt auf dem Linux Samba Server ausgeführt werden. Nur wie kommen die Updates auf die Clients? Wie immer lautet die Antwort: Per Ansible Playbook:</p> <div class="highlight"><pre><span></span><span class="nt">---</span> <span class="nt">-</span> <span class="nt">name</span><span class="o">:</span> <span class="nt">mount</span> <span class="nt">updates</span> <span class="nt">share</span> <span class="nt">and</span> <span class="nt">run</span> <span class="nt">update</span> <span class="nt">raw</span><span class="o">:</span> <span class="s2">&quot;net use U: \\\\meinsambaserver\\updates /persistent:no; cmd /c U:\\wsusoffline\\client\\cmd\\DoUpdate.cmd&quot;</span> <span class="nt">register</span><span class="o">:</span> <span class="nt">command_result</span> <span class="nt">changed_when</span><span class="o">:</span> <span class="s2">&quot;&#39;Installation successful&#39; in command_result.stdout&quot;</span> <span class="nt">failed_when</span><span class="o">:</span> <span class="nt">-</span> <span class="s2">&quot;&#39;Nothing to do!&#39; not in command_result.stdout&quot;</span> <span class="nt">-</span> <span class="s2">&quot;&#39;Installation successful.&#39; not in command_result.stdout&quot;</span> </pre></div> <p>WSUS Offline liegt dabei auf einem <code>Updates</code>-Share. Die zwei schwierigsten Sachen war das Escaping der Orte für den Kommandoaufruf und den richtigen Status erkennen. Aus irgendeinem Grund hat WSUS Offline immer den Return Code 1 raus wenn er was installiert hat. Mit der Hilfe von <code>changed_when</code> und <code>failed_when</code> suche ich in dem STDOUT-Output nach bestimmten Stichworten um den richtigen Status zu bekommen. Bestimmt decke ich nicht alles ab, aber in meinen ersten Tests funktioniert es.</p>Irgendwas mit Bleistiften2016-05-09T13:43:00+02:002016-05-09T13:43:00+02:00marvintag:xsteadfastx.org,2016-05-09:/2016/05/09/irgendwas-mit-bleistiften/<p><a href="http://giphy.com/gifs/interesting-pencils-mtqTjXCMg8DaU"><img src="http://media0.giphy.com/media/mtqTjXCMg8DaU/giphy.gif" alt="source: http://www.reddit.com/r/interestingasfuck/comments/31ivk6/sharpening_pencils/"></a></p> <p>Sorry. Ich konnte die letzte Zeit nicht bloggen. Ich hatte keine Zeit. Ich musste mich in das Thema Bleistifte reinnerden. Allein dieser Post ist seit über einer Woche im Texteditor geöffnet. Ich mag es mich in neue Themen einzufinden. Und dann kann es vorkommen das es mich total überkommt und …</p><p><a href="http://giphy.com/gifs/interesting-pencils-mtqTjXCMg8DaU"><img src="http://media0.giphy.com/media/mtqTjXCMg8DaU/giphy.gif" alt="source: http://www.reddit.com/r/interestingasfuck/comments/31ivk6/sharpening_pencils/"></a></p> <p>Sorry. Ich konnte die letzte Zeit nicht bloggen. Ich hatte keine Zeit. Ich musste mich in das Thema Bleistifte reinnerden. Allein dieser Post ist seit über einer Woche im Texteditor geöffnet. Ich mag es mich in neue Themen einzufinden. Und dann kann es vorkommen das es mich total überkommt und ich mit Haut und Haaren gefangen bin und alles in mich aufsage wie ein sehr trockener Schwamm. Seitdem fräse ich mich durch diverse Bleistifte und Spitzer. Auf meinem Schreibtisch thront schon ein kleines Bleistift Arsenal. Ihr wisst gar nicht wie schwer es ist einen guten Anspitzer zu finden der im Rucksack nicht rumsaut.</p> <p>Seit Anfang des Jahres habe ich mich dazu entschlossen wieder Tagebuch zu führen. Nichts besonders. Es geht darum endlich wieder zu lernen Gedanken aufzuschreiben. Als Teenager schreibt man viel auf und es ist einem ziemlich egal wie es aussieht oder wirkt. Als ich anfing ins Internet zu schreiben, wurde mir meine Reschtschreibschwäche zum Verhängnis. Menschen aus meinem Umfeld fingen an Witze darüber zu machen. Nichts wirklich schlimmes, trotzdem reichte es das ich nur sehr ungern schrieb und falls ich mal etwas vergassen musste, ich mich kaum traute nochmal drüber zu lesen. Jemand Wichtiges sagt mir einmal, es ist egal wie es aussieht... Hauptsache man schreibt. Und seit dem fülle ich fast jeden Tag mein Notizbuch mit Zeugs. Und das am liebstem mit einem Bleistift.</p> <p>Und was tut man da so wenn man Langeweile hat? Man googelt nach Bleistift Blogs und Podcasts. Hier also ein paar Links zum Einsteigen:</p> <p>Zur Hausarbeit höre ich die letzte Zeit den <a href="http://www.erasable.us/">Erasable</a> Podcast nach. Auch wenn es mir schwer fällt als Neueinsteiger allem zu folgen, mag ich die spürbare Begeisterung.</p> <p>Das <a href="http://www.lexikaliker.de/">Lexikaliker</a> Blog habe ich von Anfang bis Ende durchgesehen. Unfassbares Know-How und so viele Information die in einigen Bestellungen geendet sind.</p> <p>Das gleiche gilt für das <a href="https://contrapuntalism.wordpress.com/">Contrapuntalism</a> Blog. Hier schreibt der Autor unter anderem über seine Besuche in Stein bei Faber-Castell. Dadurch habe ich gelernt das ich für 5 Jahre mitten im Schmelztigel der Bleistiftherstellung gelebt habe und es leider nicht bemerkt habe.</p> <p>Bald wird es auch eine Dokumentation über Bleistifte geben. Der Name <a href="https://filmfreeway.com/project/421396">No. 2: Story of the Pencil</a>. Leider kann man ihn noch nicht sehen. Hier der Trailer:</p> <p><span class="videobox"> <iframe src="//player.vimeo.com/video/132680043?title=0&amp;byline=0&amp;portrait=0" width="640" height="390" frameborder="0" webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p>Ein Teilnehmer des Erasable Podcasts schreibt auf <a href="http://www.pencilrevolution.com/">Pencil Revolution</a>.</p> <p>Und dann gibt es da noch einen einen <a href="http://cwpencils.com/">Laden</a> in New York der wirklich nur Bleistifte verkauft.</p> <p>Also erwartet bald ein mehr Bleistift Content.</p>Prince.2016-04-25T14:45:00+02:002016-04-25T14:45:00+02:00marvintag:xsteadfastx.org,2016-04-25:/2016/04/25/prince/<p><a href="http://www.newyorker.com/culture/culture-desk/cover-story-2016-05-02"><img alt="Das Cover des New Yorkers vom 2. Mai 2016" src="http://xsteadfastx.org/images/purple-rain.jpg"></a></p> <p>Prince ist tot und das Jahr 2016 zieht weiter mordend durch den Raum um noch mehr der größten Künstler zu rauben. Als ich 1984 geboren wurde, sagte mein Onkel das dies ein gutes Jahr sei, meine Geburt und "Purple Rain". Das eine war der Beginn meiner eigenen Zeit und das …</p><p><a href="http://www.newyorker.com/culture/culture-desk/cover-story-2016-05-02"><img alt="Das Cover des New Yorkers vom 2. Mai 2016" src="http://xsteadfastx.org/images/purple-rain.jpg"></a></p> <p>Prince ist tot und das Jahr 2016 zieht weiter mordend durch den Raum um noch mehr der größten Künstler zu rauben. Als ich 1984 geboren wurde, sagte mein Onkel das dies ein gutes Jahr sei, meine Geburt und "Purple Rain". Das eine war der Beginn meiner eigenen Zeit und das andere das Release eines der vielleicht besten Alben aller Zeiten. Prince war immer ein großer Teil meiner ersten musikalischen Wahrnehmungen. Mein Onkel erzählte mir pausenlos von ihm. Überall in seinem Zimmer hingen Poster von Prince. Es war der erste Baustein zu meiner sich entwickelnden Musikliebe. Nun sitze ich in einem ähnlichen Zimmer. Eine kleine Reproduktion seines damaligen Zimmers. Meine Schränke sind gefüllt mit seinen Platten die er mir vermacht hat. Der größte Ritterschlag, damit beauftragt zu werden diese Platten aufzubewahren und sie zu spielen. Schließlich steckt da ein halbes Leben drin. All die biografischen Verbindungen. Er erzähle von den Prince Konzerten die er besucht hatte. Das eine mal als die Menge an Menschen ihm die Schuhe ausgezogen hat, als sie ihn, einer Meeresbrandung gleich, vor und zurück trieben ließen und er nur in Socken das Konzert verlassen musste. Natürlich erst nach der letzten Zugabe. Oder das eine mal als er die Sonnenbrille von Prince fing die er ins Publikum geschleudert hat. Für mich klang das immer nach den besten Erlebnissen die man als Erwachsener erleben kann. Und auch heute hat sich meine Meinung kaum geändert. Er erzählte mir wie er einmal auf die geheime After Show Party in London gelangt ist. Prince saß auf der Bühne im dunkeln, nur er und seine Gitarre und ein Gitarren-Soli das eine halbe Stunde dauerte. Dies ließ für mich den musikalischen Kontext von Prince ins unermäßliche wachsen. Ich habe für die Musik von Prince, und sein faszinierendes absurdes Leben, sehr viel übrig. Doch die größte Faszination für Prince ist die meines Onkel und was sie für einen Eindruck bei mir hinterlassen hat. Etwas scheinbar religiöses. Etwas das sein Leben ausmacht. Als ich heute von dem Tod von Prince las, schrieb ich meinem Onkel eine Nachricht wie Leid es mir tut, danach legte ich "Controversy" auf und dachte daran wie sehr er mir gezeigt hat wie groß Musik wirklich sein kann.</p>Alone in Paris2016-04-20T13:10:00+02:002016-04-20T13:10:00+02:00marvintag:xsteadfastx.org,2016-04-20:/2016/04/20/alone-in-paris/<p>Einmal Paris alleine erleben, ohne Touristen. Ein Paradox wenn man selber einer ist.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/Syy1vZ09sv8' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p><p>Einmal Paris alleine erleben, ohne Touristen. Ein Paradox wenn man selber einer ist.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/Syy1vZ09sv8' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Dokumentation: Traumhaus Baumhaus2016-04-18T10:27:00+02:002016-04-18T10:27:00+02:00marvintag:xsteadfastx.org,2016-04-18:/2016/04/18/dokumentation-traumhaus-baumhaus/<p>Auf Arte ist immer verlass. Sonntag hatte ich lust auf eine Dokumentation und scrollte durch die Arte App nach etwas passendem. Beim Titel <a href="http://www.arte.tv/guide/de/058360-000-A/traumhaus-baumhaus">Traumhaus Baumhaus</a> blieb ich natürlich hängen. Baumhäuse sind hier immer Ontopic. Was fehlt ist <a href="http://thecindercone.com/">The Conder Cone</a>.</p> <blockquote> <p>In der Auvergne im Wald von La Stèle haben Thomas …</p></blockquote><p>Auf Arte ist immer verlass. Sonntag hatte ich lust auf eine Dokumentation und scrollte durch die Arte App nach etwas passendem. Beim Titel <a href="http://www.arte.tv/guide/de/058360-000-A/traumhaus-baumhaus">Traumhaus Baumhaus</a> blieb ich natürlich hängen. Baumhäuse sind hier immer Ontopic. Was fehlt ist <a href="http://thecindercone.com/">The Conder Cone</a>.</p> <blockquote> <p>In der Auvergne im Wald von La Stèle haben Thomas und Mathieu ihren Traum wahr werden lassen: Sie haben ein ganzes Baumhaus-Dorf angelegt. Die Baumhäuser sind so konzipiert, dass die Bewohner von ihrer Unterkunft in bis zu 13 Metern Höhe keine Sicht auf die anderen Häuser haben. So können sie sich ganz allein im Tannenwald fühlen. -- <cite><a href="http://www.arte.tv/guide/de/058360-000-A/traumhaus-baumhaus">Arte</a></cite></p> </blockquote> <iframe src="http://www.arte.tv/guide/de/embed/058360-000-A/medium" allowfullscreen="true" style="width: 600px; height: 344px;" frameborder="0"></iframe>Erster Teaser zu A Killing Joke2016-04-15T14:00:00+02:002016-04-15T14:00:00+02:00marvintag:xsteadfastx.org,2016-04-15:/2016/04/15/erster-teaser-zu-a-killing-joke/<p>Ich mag die animierten Comic-Verfilmungen von DC. Wenn ich darüber nachdenke welcher Batman mich am meisten geprägt hat werde ich immer antworten: Der Batman aus der <a href="https://de.wikipedia.org/wiki/Batman_(Zeichentrickserie)">animierten Serie</a>. Ein Highlight wenn ich aus der Schule kam und Abends dann eine Episode ausgestrahlt wurde. Meine Actionfigur aus der Serie ist vielleicht …</p><p>Ich mag die animierten Comic-Verfilmungen von DC. Wenn ich darüber nachdenke welcher Batman mich am meisten geprägt hat werde ich immer antworten: Der Batman aus der <a href="https://de.wikipedia.org/wiki/Batman_(Zeichentrickserie)">animierten Serie</a>. Ein Highlight wenn ich aus der Schule kam und Abends dann eine Episode ausgestrahlt wurde. Meine Actionfigur aus der Serie ist vielleicht mein liebstes Spielzeug gewesen. Meine eigenen und ziemlich harmlosen Abenteuer, das Wohnzimmer als eine nicht so düstere Version von Gotham City. Sehr zu empfehlen ist der Film <a href="https://de.wikipedia.org/wiki/Batman_und_das_Phantom">Mask of the Phantasm</a>. Vielleicht sogar der beste Batman Film.</p> <p>Zu den düstersten Batman Comics die ich gelesen habe, zählt <a href="https://de.wikipedia.org/wiki/Batman:_Lächeln,_bitte!">A Killing Joke</a>. Nicht zuletzt wegen einer bestimmten Szene (keine Spoiler). Nun wurde aus diesem Meisterwerk von <a href="https://de.wikipedia.org/wiki/Alan_Moore">Alan Moore</a> ein animierter Film. Hier ein erster Teaser.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/MnJG79xCofE' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Slowdive - Souvlaki Space Station2016-04-13T15:49:00+02:002016-04-13T15:49:00+02:00marvintag:xsteadfastx.org,2016-04-13:/2016/04/13/slowdive-souvlaki-space-station/<p>Ein wenig Shoegaze-Geschichte am Mittwoch Nachmittag. Gestern die Hoffnungen auf das Champions League Halbfinale verspielt und mit leichter Melancholie vom Blickwinkel aus des ersten Jahresviertels heute aufgewacht. Da ist Slowdive ein wenig Medizin.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/tqkXqHTNPBg' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p><p>Ein wenig Shoegaze-Geschichte am Mittwoch Nachmittag. Gestern die Hoffnungen auf das Champions League Halbfinale verspielt und mit leichter Melancholie vom Blickwinkel aus des ersten Jahresviertels heute aufgewacht. Da ist Slowdive ein wenig Medizin.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/tqkXqHTNPBg' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Ansible, win_package und Upgrades2016-04-12T12:15:00+02:002016-04-12T12:15:00+02:00marvintag:xsteadfastx.org,2016-04-12:/2016/04/12/ansible-win-package-und-upgrades/<p><a href="http://giphy.com/gifs/transparent-supernatural-b5TmEXrjZ2II8"><img src="http://media2.giphy.com/media/b5TmEXrjZ2II8/giphy.gif" alt="source: http://childofmoonlight.tumblr.com/post/44755113585/im-a-grumpy-guss-enjoy-this-gif-set-of-grumpy"></a></p> <p>Das Ansible Modul <a href="https://docs.ansible.com/ansible/win_package_module.html">win_package</a> beruht auf die Annahme das die <code>product_id</code> in der Registry vorhanden ist oder eben nicht. Davon macht es abhängig ob ein Paket installiert werden soll oder ob es schon vorhanden ist. Nun kann es ja auch vorkommen das man ein Paket, obwohl es laut Registry schon …</p><p><a href="http://giphy.com/gifs/transparent-supernatural-b5TmEXrjZ2II8"><img src="http://media2.giphy.com/media/b5TmEXrjZ2II8/giphy.gif" alt="source: http://childofmoonlight.tumblr.com/post/44755113585/im-a-grumpy-guss-enjoy-this-gif-set-of-grumpy"></a></p> <p>Das Ansible Modul <a href="https://docs.ansible.com/ansible/win_package_module.html">win_package</a> beruht auf die Annahme das die <code>product_id</code> in der Registry vorhanden ist oder eben nicht. Davon macht es abhängig ob ein Paket installiert werden soll oder ob es schon vorhanden ist. Nun kann es ja auch vorkommen das man ein Paket, obwohl es laut Registry schon installiert ist, es noch einmal installieren möchte. Quasi ein Upgrade machen. Schön wäre es wenn er nicht nur schaut ob das Paket installiert ist, sondern auch die installierte Version. Daran könnte man Task Entscheidungen treffen. Dies mache ich nun manuell. Ein Beispiel für VLC:</p> <div class="highlight"><pre><span></span>--- # Der ganz normale Install-Task. Es wird nach der product_id gesucht gegebenenfalls installiert - name: install win_package: product_id=&quot;VLC media player&quot; path=&quot;//myserver/updates/software/vlc/vlc-2.2.2-win32.exe&quot; arguments=&quot;/L=1031 /S&quot; # Ein Powershell Snippet das die Version des installierten Pakets in der Variabel &quot;version&quot; speichert - name: check version raw: (Get-ItemProperty &quot;HKLM:\SOFTWARE\wow6432node\Microsoft\Windows\CurrentVersion\Uninstall\VLC media player&quot;).DisplayVersion register: version - name: upgrade win_package: product_id=&quot;VLC media player upgrade&quot; # Muss anders sein damit das Paket nochmal installiert wird path=&quot;//myserver/updates/software/vlc/vlc-2.2.2-win32.exe&quot; arguments=&quot;/L=1031 /S&quot; register: upgrade_results # Speichert das Ergebnis des Tasks in die Variabel &quot;upgrade_results&quot; changed_when: &#39;&quot;was installed&quot; in upgrade_results.msg&#39; # Ändert den Status auf &quot;changed&quot; wenn der String &quot;was installed&quot; im Ergebnis ist failed_when: &#39;&quot;was installed&quot; not in upgrade_results.msg&#39; # Status &quot;failed&quot; wenn &quot;was installed&quot; nicht im Ergebnis ist ignore_errors: True # Sonst bricht er ab bevor der Task überhaupt die Variabel &quot;upgrade_results&quot; befüllt when: &#39;&quot;2.2.2&quot; not in version.stdout_lines&#39; # Den Task nur ausführen wenn die Version eine andere ist </pre></div>Paris in Bildern 20162016-04-11T15:28:00+02:002016-04-11T15:28:00+02:00marvintag:xsteadfastx.org,2016-04-11:/2016/04/11/paris-in-bildern-2016/<p>Eigentlich wollte ich so viel schreiben. Das habe ich auch. Nur weiß ich nicht ob die Sachen jemals mein Notizbuch verlassen werden. Also erstmal ein paar Eindrücke, gebannt auf Ilford HP5... mit wenigen Ausnahmen.</p> <p><a href="https://www.flickr.com/photos/8810721@N07/26272005452/"><img alt="Rodin" src="https://farm2.staticflickr.com/1531/26272005452_7bd8175256_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25761633043/"><img alt="Rodin" src="https://farm2.staticflickr.com/1548/25761633043_a703e1099a_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26364375095/"><img alt="Paris" src="https://farm2.staticflickr.com/1485/26364375095_e6bff59174_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26091495910/"><img alt="Paris" src="https://farm2.staticflickr.com/1645/26091495910_babf3f132a_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26169928481/"><img alt="IMG_20160323_195211-01" src="https://farm2.staticflickr.com/1506/26169928481_cac0917715_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/25631465354/"><img alt="IMG_20160323_211441-01" src="https://farm2.staticflickr.com/1548/25631465354_6bae41e4d9_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25761607033/"><img alt="Paris" src="https://farm2.staticflickr.com/1560/25761607033_09174223dc_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26091497180/"><img alt="Rodin" src="https://farm2.staticflickr.com/1689/26091497180_f2611ed883_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26210307296/"><img alt="IMG_20160324_120306-01" src="https://farm2.staticflickr.com/1671/26210307296_5b895ae4e1_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26210310106/"><img alt="IMG_20160324_121516-01" src="https://farm2.staticflickr.com/1530/26210310106_da726dacbc_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26194786912/"><img alt="IMG_20160325_214209" src="https://farm2.staticflickr.com/1492/26194786912_0e24d89bbf_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25759512694/"><img alt="Rodin" src="https://farm2.staticflickr.com/1518/25759512694_292964bd5f_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26338413576/"><img alt="Rodin" src="https://farm2.staticflickr.com/1495/26338413576_01776a97f8_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26271964152/"><img alt="Institut Finlandais" src="https://farm2.staticflickr.com/1715/26271964152_30f448a055_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26091481970/"><img alt="Paris" src="https://farm2.staticflickr.com/1701/26091481970_5c0f9331e7_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25759497224/"><img alt="Paris" src="https://farm2.staticflickr.com/1469/25759497224_1c450a131a_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26298097331/"><img alt="Paris" src="https://farm2.staticflickr.com/1471/26298097331_bb1df881bb_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/25633528013/"><img alt="IMG_20160321_144859-01" src="https://farm2.staticflickr.com/1634/25633528013_6ceff7c3ab_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26338384956/"><img alt="Paris" src="https://farm2.staticflickr.com/1628/26338384956_b517eeb350_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26298029591/"><img alt="Paris" src="https://farm2.staticflickr.com/1584/26298029591_df34a86755_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26364267055/"><img alt="Paris" src="https://farm2.staticflickr.com/1441/26364267055_c8420003af_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/25633544833/"><img alt="IMG_20160323_135136-01" src="https://farm2.staticflickr.com/1448/25633544833_be7219f532_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26271900472/"><img alt="Paris" src="https://farm2.staticflickr.com/1670/26271900472_a72897e5c0_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26364269115/"><img alt="Paris" src="https://farm2.staticflickr.com/1454/26364269115_1058177d16_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26364270355/"><img alt="Paris" src="https://farm2.staticflickr.com/1639/26364270355_b7e6d83c6f_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26298014601/"><img alt="Paris" src="https://farm2.staticflickr.com/1590/26298014601_ecaf710b88_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26194778242/"><img alt="IMG_20160320_230959" src="https://farm2.staticflickr.com/1501/26194778242_b3bff4509c_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26014344450/"><img alt="IMG_20160320_232537" src="https://farm2.staticflickr.com/1622/26014344450_7409dc5365_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26271884742/"><img alt="Paris" src="https://farm2.staticflickr.com/1626/26271884742_271381d5f6_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26298016031/"><img alt="Paris" src="https://farm2.staticflickr.com/1670/26298016031_4619ba6aac_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/25963316200/"><img alt="IMG_20160323_121429-01" src="https://farm2.staticflickr.com/1580/25963316200_a419fe204c_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25759417384/"><img alt="Paris" src="https://farm2.staticflickr.com/1510/25759417384_544b1322de_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25759418134/"><img alt="Paris" src="https://farm2.staticflickr.com/1547/25759418134_5851609125_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26297998741/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1598/26297998741_0a3a27cb2e_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26297981971/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1614/26297981971_ba5094d626_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26091362070/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1444/26091362070_5e4bfcce9e_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26091358750/"><img alt="Paris" src="https://farm2.staticflickr.com/1636/26091358750_61df1245de_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26210281696/"><img alt="IMG_20160321_224531-01" src="https://farm2.staticflickr.com/1527/26210281696_490b0760de_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26338268666/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1680/26338268666_3fb4ce995d_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25761465323/"><img alt="Paris" src="https://farm2.staticflickr.com/1481/25761465323_31fea34546_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25759368834/"><img alt="Paris" src="https://farm2.staticflickr.com/1543/25759368834_7f53c29b75_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26271840182/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1612/26271840182_f7236e2bd0_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26364209895/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1665/26364209895_02a0b82826_b.jpg"></a></p><p>Eigentlich wollte ich so viel schreiben. Das habe ich auch. Nur weiß ich nicht ob die Sachen jemals mein Notizbuch verlassen werden. Also erstmal ein paar Eindrücke, gebannt auf Ilford HP5... mit wenigen Ausnahmen.</p> <p><a href="https://www.flickr.com/photos/8810721@N07/26272005452/"><img alt="Rodin" src="https://farm2.staticflickr.com/1531/26272005452_7bd8175256_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25761633043/"><img alt="Rodin" src="https://farm2.staticflickr.com/1548/25761633043_a703e1099a_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26364375095/"><img alt="Paris" src="https://farm2.staticflickr.com/1485/26364375095_e6bff59174_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26091495910/"><img alt="Paris" src="https://farm2.staticflickr.com/1645/26091495910_babf3f132a_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26169928481/"><img alt="IMG_20160323_195211-01" src="https://farm2.staticflickr.com/1506/26169928481_cac0917715_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/25631465354/"><img alt="IMG_20160323_211441-01" src="https://farm2.staticflickr.com/1548/25631465354_6bae41e4d9_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25761607033/"><img alt="Paris" src="https://farm2.staticflickr.com/1560/25761607033_09174223dc_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26091497180/"><img alt="Rodin" src="https://farm2.staticflickr.com/1689/26091497180_f2611ed883_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26210307296/"><img alt="IMG_20160324_120306-01" src="https://farm2.staticflickr.com/1671/26210307296_5b895ae4e1_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26210310106/"><img alt="IMG_20160324_121516-01" src="https://farm2.staticflickr.com/1530/26210310106_da726dacbc_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26194786912/"><img alt="IMG_20160325_214209" src="https://farm2.staticflickr.com/1492/26194786912_0e24d89bbf_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25759512694/"><img alt="Rodin" src="https://farm2.staticflickr.com/1518/25759512694_292964bd5f_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26338413576/"><img alt="Rodin" src="https://farm2.staticflickr.com/1495/26338413576_01776a97f8_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26271964152/"><img alt="Institut Finlandais" src="https://farm2.staticflickr.com/1715/26271964152_30f448a055_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26091481970/"><img alt="Paris" src="https://farm2.staticflickr.com/1701/26091481970_5c0f9331e7_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25759497224/"><img alt="Paris" src="https://farm2.staticflickr.com/1469/25759497224_1c450a131a_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26298097331/"><img alt="Paris" src="https://farm2.staticflickr.com/1471/26298097331_bb1df881bb_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/25633528013/"><img alt="IMG_20160321_144859-01" src="https://farm2.staticflickr.com/1634/25633528013_6ceff7c3ab_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26338384956/"><img alt="Paris" src="https://farm2.staticflickr.com/1628/26338384956_b517eeb350_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26298029591/"><img alt="Paris" src="https://farm2.staticflickr.com/1584/26298029591_df34a86755_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26364267055/"><img alt="Paris" src="https://farm2.staticflickr.com/1441/26364267055_c8420003af_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/25633544833/"><img alt="IMG_20160323_135136-01" src="https://farm2.staticflickr.com/1448/25633544833_be7219f532_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26271900472/"><img alt="Paris" src="https://farm2.staticflickr.com/1670/26271900472_a72897e5c0_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26364269115/"><img alt="Paris" src="https://farm2.staticflickr.com/1454/26364269115_1058177d16_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26364270355/"><img alt="Paris" src="https://farm2.staticflickr.com/1639/26364270355_b7e6d83c6f_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26298014601/"><img alt="Paris" src="https://farm2.staticflickr.com/1590/26298014601_ecaf710b88_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26194778242/"><img alt="IMG_20160320_230959" src="https://farm2.staticflickr.com/1501/26194778242_b3bff4509c_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26014344450/"><img alt="IMG_20160320_232537" src="https://farm2.staticflickr.com/1622/26014344450_7409dc5365_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26271884742/"><img alt="Paris" src="https://farm2.staticflickr.com/1626/26271884742_271381d5f6_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26298016031/"><img alt="Paris" src="https://farm2.staticflickr.com/1670/26298016031_4619ba6aac_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/25963316200/"><img alt="IMG_20160323_121429-01" src="https://farm2.staticflickr.com/1580/25963316200_a419fe204c_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25759417384/"><img alt="Paris" src="https://farm2.staticflickr.com/1510/25759417384_544b1322de_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25759418134/"><img alt="Paris" src="https://farm2.staticflickr.com/1547/25759418134_5851609125_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26297998741/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1598/26297998741_0a3a27cb2e_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26297981971/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1614/26297981971_ba5094d626_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26091362070/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1444/26091362070_5e4bfcce9e_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26091358750/"><img alt="Paris" src="https://farm2.staticflickr.com/1636/26091358750_61df1245de_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26210281696/"><img alt="IMG_20160321_224531-01" src="https://farm2.staticflickr.com/1527/26210281696_490b0760de_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26338268666/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1680/26338268666_3fb4ce995d_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25761465323/"><img alt="Paris" src="https://farm2.staticflickr.com/1481/25761465323_31fea34546_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/25759368834/"><img alt="Paris" src="https://farm2.staticflickr.com/1543/25759368834_7f53c29b75_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26271840182/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1612/26271840182_f7236e2bd0_b.jpg"></a></p> <p><a href="https://www.flickr.com/photos/8810721@N07/26364209895/"><img alt="Panthéon" src="https://farm2.staticflickr.com/1665/26364209895_02a0b82826_b.jpg"></a></p>Ansible, win_package und die product_id2016-04-08T11:55:00+02:002016-04-08T11:55:00+02:00marvintag:xsteadfastx.org,2016-04-08:/2016/04/08/ansible-win-package-und-die-product-id/<p>Gerade läuft das letzte Windows 7 auf Windows 10 Update. Diese Möglichkeit habe ich genutzt um für die Administration der Clients von <a href="http://saltstack.com">SaltStack</a> zu <a href="http://ansible.com">Ansible</a> zu wechseln. Windows Administration ist für mich eher so ein leidiges Thema, was mich ziemlich oft auf die Palme bringt. Ansible benutze ich auf den …</p><p>Gerade läuft das letzte Windows 7 auf Windows 10 Update. Diese Möglichkeit habe ich genutzt um für die Administration der Clients von <a href="http://saltstack.com">SaltStack</a> zu <a href="http://ansible.com">Ansible</a> zu wechseln. Windows Administration ist für mich eher so ein leidiges Thema, was mich ziemlich oft auf die Palme bringt. Ansible benutze ich auf den Linux Servern für fast alles. Mal schnell ein paar Update einspielen oder auch für das ausrollen meines <a href="https://github.com/xsteadfastx/batcave">batcaves</a>.</p> <h2>Master vorbereiten</h2> <p>Neben Ansible sollte man noch das Modul <code>pywinrm</code> installieren:</p> <p><code>pip install "pywinrm&gt;=0.1.1"</code></p> <h2>Clients vorbereiten</h2> <p>Um Ansible unter Windows zu benutzen muss ein <a href="https://github.com/ansible/ansible/blob/devel/examples/scripts/ConfigureRemotingForAnsible.ps1">PowerShell Script</a> ausgeführt werden. Damit dies funktioniert muss erst eine PowerShell Policy angepasst werden. Dies tut man mit <code>set-executionpolicy remotesigned</code>. Danach Script ausführen und alles sollte eingerichtet sein.</p> <h2>Inventory</h2> <p>Auch die hosts-Datei muss ein wenig angepasst werden. Es werden ein paar zusätzliche Daten benötigt. Wenn das hosts-File so aussieht:</p> <div class="highlight"><pre><span></span><span class="k">[windows]</span> <span class="na">192.168.1.5</span> <span class="na">192.168.1.6</span> <span class="na">192.168.1.7</span> </pre></div> <p>habe ich eine <code>group_vars/windows.yml</code> angelegt die so aussieht:</p> <div class="highlight"><pre><span></span><span class="n">ansible_user</span><span class="o">:</span> <span class="n">winadminuser</span> <span class="n">ansible_password</span><span class="o">:</span> <span class="n">winadminpassword</span> <span class="n">ansible_port</span><span class="o">:</span> <span class="mi">5986</span> <span class="n">ansible_connection</span><span class="o">:</span> <span class="n">winrm</span> <span class="n">ansible_winrm_server_cert_validation</span><span class="o">:</span> <span class="n">ignore</span> </pre></div> <p>Dieses File kann man mit <code>ansible-vault encrypt group_vars/windows.yml</code> verschlüßeln.</p> <h2>Der erste Test</h2> <p><code>ansible windows -m win_ping --ask-vault-pass</code> sollte für einen ersten Versuch reichen.</p> <h2>Pakete installieren</h2> <p>Hier ein einfaches Beispiel um ein ein Paket zu installieren:</p> <div class="highlight"><pre><span></span>--- - name: Install the vc thingy win_package: name=&quot;Microsoft Visual C thingy&quot; path=&quot;http://download.microsoft.com/download/1/6/B/16B06F60-3B20-4FF2-B699-5E9B7962F9AE/VSU_4/vcredist_x64.exe&quot; Product_Id=&quot;{CF2BEA3C-26EA-32F8-AA9B-331F7E34BA97}&quot; Arguments=&quot;/install /passive /norestart&quot; </pre></div> <p>Ich hatte mit fast allen Paketen Probleme die <code>product_id</code> herauszufinden. Erst dachte ich es handelt sich hierbei um den Titel der Software in den Systemeinstellungen unter Windows. Pustekuchen. Man muss sich diese ID aus den Registry popeln. Diese befindet sich unter <code>HKLM:\Software\microsoft\windows\currentversion\uninstall</code>. Aber die sah bei mir ziemlich dünn aus und war bei weiten nicht mit allen Sachen gefüllt die ich installiert hatte. Stellt sich herraus das ein Teil der Registry sich unter <code>HKLM:\Software\wow6432node\microsoft\windows\currentversion\uninstall</code> versteckt. Und zwar wenn es sich um 32bit Pakete handelt.</p>Neue Dag Nasty Single2016-04-05T10:32:00+02:002016-04-05T10:32:00+02:00marvintag:xsteadfastx.org,2016-04-05:/2016/04/05/neue-dag-nasty-single/<p><img alt="Cover" src="http://xsteadfastx.org/images/dag-nasty-cold-heart-wanting-nothing.jpg"></p> <p>Dag Nasty's "Can I Say" ist das Album welches mir seit Jahren Ohrwürmer erzeugt. Gerade wieder seit Wochen der Song <a href="https://www.youtube.com/watch?v=sOO0tZoD4aQ">One to Two</a>. Nun lese ich zufällig das sie zwei neue Songs aufgenommen haben und diese Ende Mai erscheinen. Leider gibt es nur zwei Soundclips aber die reichen mir schon …</p><p><img alt="Cover" src="http://xsteadfastx.org/images/dag-nasty-cold-heart-wanting-nothing.jpg"></p> <p>Dag Nasty's "Can I Say" ist das Album welches mir seit Jahren Ohrwürmer erzeugt. Gerade wieder seit Wochen der Song <a href="https://www.youtube.com/watch?v=sOO0tZoD4aQ">One to Two</a>. Nun lese ich zufällig das sie zwei neue Songs aufgenommen haben und diese Ende Mai erscheinen. Leider gibt es nur zwei Soundclips aber die reichen mir schon um mich völlig anzufixen.</p> <blockquote> <p>Two songs by Dag Nasty recorded with Ian MacKaye and Don Zientara at Inner Ear Studios in December, 2015. For this session, the band returned to its original lineup: guitarist Brian Baker, singer Shawn Brown, bassist Roger Marbury, and drummer Colin Sears.</p> </blockquote> <p><iframe width="100%" height="400" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?visual=true&url=http%3A%2F%2Fapi.soundcloud.com%2Ftracks%2F256532997&show_artwork=true"></iframe></p>Pariser Pinholefotos2016-04-04T13:32:00+02:002016-04-04T13:32:00+02:00marvintag:xsteadfastx.org,2016-04-04:/2016/04/04/pariser-pinholefotos/<p><a href="https://www.flickr.com/photos/marvinxsteadfast/26228757855/"><img src="https://farm2.staticflickr.com/1441/26228757855_b642e03abb_b.jpg" alt="Paris"></a></p> <p>Nach unserem Umzug hätte ich nicht mehr damit gerechnet meine <a href="https://filmphotography.eu/kamera/holga-120-wide-pinhole/">Holga WPC 120</a> wieder in meinen Händen zuhalten. Und wenn sie wieder aufgetaucht wäre, hätte ich bestimmt unseren Aufenthalt in Paris damit verpasst. Vor jedem Ausflug stellt man sich die essentielle Frage nach dem perfekten Kamera-Setup. Am liebsten beschränke ich …</p><p><a href="https://www.flickr.com/photos/marvinxsteadfast/26228757855/"><img src="https://farm2.staticflickr.com/1441/26228757855_b642e03abb_b.jpg" alt="Paris"></a></p> <p>Nach unserem Umzug hätte ich nicht mehr damit gerechnet meine <a href="https://filmphotography.eu/kamera/holga-120-wide-pinhole/">Holga WPC 120</a> wieder in meinen Händen zuhalten. Und wenn sie wieder aufgetaucht wäre, hätte ich bestimmt unseren Aufenthalt in Paris damit verpasst. Vor jedem Ausflug stellt man sich die essentielle Frage nach dem perfekten Kamera-Setup. Am liebsten beschränke ich mich auf eine Kamera und wenn es nicht anders geht wird noch eine zweite mitgenommen. Alles darüber ist soviel Balast für mich, körperlich aber auch durch duch den lähmenden fotografischen Druck zu versagen. Also nahm ich meine Leica Mini 2 mit und die besagte Holga WPC 120, eine Pinholekamera.</p> <p>Gleich beim ersten Foto kam es zu einer Doppelbelichtung die im Endprodukt keinen Einfluss auf das Bild hatte. Ich hatte vergessen das ich auf dem 120er Film immer zwei Bilder weiterdrehen muss bei dieser Kamera. Da das zweite Bild eine Innenaufnhame des focaultschen Pendel im Pantheon war, war das Bild so unterbelichtet das es das andere nicht beeinträchtigt hat.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26162453681/"><img src="https://farm2.staticflickr.com/1482/26162453681_0d88533bcd_b.jpg" alt="Paris"></a></p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26228745735/"><img src="https://farm2.staticflickr.com/1491/26228745735_4289e74ee5_b.jpg" alt="Paris"></a></p> <p>Diese beiden Bilder habe ich im <a href="https://de.wikipedia.org/wiki/Jardin_du_Luxembourg">Jardin du Luxembourg</a> aufgenommen. Auf dem letzteren befindet sich unser liebster Baum in diesem wunderschönen Park.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26228741755/"><img src="https://farm2.staticflickr.com/1672/26228741755_4beb918c98_b.jpg" alt="Paris"></a></p> <p>Und nochmal die Seine mit ihnren anscheinend bewohnten Hausbooten.</p> <p><a href="https://www.flickr.com/photos/marvinxsteadfast/26136279792/"><img src="https://farm2.staticflickr.com/1484/26136279792_aafc4fdf53_b.jpg" alt="Paris"></a></p> <p>Und nun eins der Kunstwerke die mich immer sehr bewegt haben. Der Denker von Rodin. Nun konnte ich das erste mal durch den Museumspark spazieren und musste gleich mein Stativ aufbauen um ein Foto zu machen.</p>WINDLICHT2016-03-18T14:06:00+01:002016-03-18T14:06:00+01:00marvintag:xsteadfastx.org,2016-03-18:/2016/03/18/windlicht/<p>Der Künstler <a href="https://en.wikipedia.org/wiki/Daan_Roosegaarde">Daan Roosegaarde</a> hat per Laser Windräder verbunden und nennt dies <a href="https://www.studioroosegaarde.net/project/windlicht/info/">WINDLICHT</a>. Wunderbar.</p> <blockquote> <p>WINDLICHT, the latest work of Roosegaarde, shows the beauty of green energy by connecting windmill blades with lines of light.</p> <p>Special software and tracking technology detect the windmill blades rotating at 280 kilometres per hour. Roosegaarde …</p></blockquote><p>Der Künstler <a href="https://en.wikipedia.org/wiki/Daan_Roosegaarde">Daan Roosegaarde</a> hat per Laser Windräder verbunden und nennt dies <a href="https://www.studioroosegaarde.net/project/windlicht/info/">WINDLICHT</a>. Wunderbar.</p> <blockquote> <p>WINDLICHT, the latest work of Roosegaarde, shows the beauty of green energy by connecting windmill blades with lines of light.</p> <p>Special software and tracking technology detect the windmill blades rotating at 280 kilometres per hour. Roosegaarde: "WINDLICHT creates the missing link between the Dutch and the beauty of our new landscape."</p> <p>Daan Roosegaarde was inspired by Kinderdijk. For him these windmills from 1740 are a perfect example of Dutch innovation. Reconnecting with the landscape and creating a positive image around green energy also drives him. Roosegaarde developed the artwork together with his team of designers and engineers.</p> </blockquote> <p><span class="videobox"> <iframe src="//player.vimeo.com/video/158649199?title=0&amp;byline=0&amp;portrait=0" width="640" height="390" frameborder="0" webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p>(<a href="http://www.nerdcore.de/2016/03/17/windlicht-laser-connected-windmills/">via</a>)</p>Python, urllib und CERTIFICATE_VERIFY_FAILED2016-03-17T13:46:00+01:002016-03-17T13:46:00+01:00marvintag:xsteadfastx.org,2016-03-17:/2016/03/17/python-urllib-und-certificate-verify-failed/<p>Da denkt man sich einfach "mal kurz mein <a href="https://github.com/xsteadfastx/blog.git">Blogging</a>-Container neu bauen" und schon funktioniert nichts mehr. Es fing schon damit an das mein <a href="https://github.com/xsteadfastx/batcave.git">Ansible Playbook</a> Probleme hatte Files herrunterzuladen. <code>urllib</code> brach immer mit <code>urlopen error [SSL: CERTIFICATE_VERIFY_FAILED]</code> ab. Wie aus dem nichts. Vor ein paar Wochen gab es keine …</p><p>Da denkt man sich einfach "mal kurz mein <a href="https://github.com/xsteadfastx/blog.git">Blogging</a>-Container neu bauen" und schon funktioniert nichts mehr. Es fing schon damit an das mein <a href="https://github.com/xsteadfastx/batcave.git">Ansible Playbook</a> Probleme hatte Files herrunterzuladen. <code>urllib</code> brach immer mit <code>urlopen error [SSL: CERTIFICATE_VERIFY_FAILED]</code> ab. Wie aus dem nichts. Vor ein paar Wochen gab es keine Probleme mit der <code>urllib</code>. Es gab sogar ein <a href="https://stackoverflow.com/questions/35762510/alpine-3-3-python-2-7-11-urllib2-causing-ssl-certificate-verify-failed">StackOverflow Posting</a> welches mir riet einfach mal die <code>ca-certificates</code> zu installieren. Das Paket war aber schon installiert. Nach viel googeln un verzweifeln bin ich auf eine <a href="http://qiita.com/runtakun/items/a7fae47bdac887db40f1">obskure Seite</a> gestoßen von der ich kaum etwas verstand. Es gab wohl die Möglichkeit <code>urllib</code> über eine Environment-Variabel den Ort der CA-Zertifikate mitzuteilen. Dies scheint auch in der <a href="https://www.python.org/dev/peps/pep-0476/#trust-database">PEP 476</a> so beschrieben. <a href="https://github.com/kennethreitz/requests/issues/2899">requests</a> scheint dies per Default zu machen. Ein <code>export SSL_CERT_FILE=/etc/ssl/certs/ca-certificates.crt</code> fixed die Situation in Alpine Linux. Ich frage mich nur was sich in ein paar Wochen so verändert hat das dieses Problem auftritt.</p>Cap'n Jazz, 7.10.952016-03-14T09:22:00+01:002016-03-14T09:22:00+01:00marvintag:xsteadfastx.org,2016-03-14:/2016/03/14/capn-jazz-7-10-95/<p>Jedes Jahr im Frühling greife ich automatisch zur Discography von Cap'n Jazz. Dies passiert ganz von alleine. Es ist nicht so das ich mir denke "Frühling Cap'n Jazz Zeit". Es war so als ich das erste mal mit 18 in meinem Auto ziellos durch die Stadt fuhr, wie auch letzte …</p><p>Jedes Jahr im Frühling greife ich automatisch zur Discography von Cap'n Jazz. Dies passiert ganz von alleine. Es ist nicht so das ich mir denke "Frühling Cap'n Jazz Zeit". Es war so als ich das erste mal mit 18 in meinem Auto ziellos durch die Stadt fuhr, wie auch letzte Woche erst in einem Ansturm von Frühlingsfreuden. Trotz 2°C weckten die Gesänge der Vogel eine Art von Euphorie. Ich scrollte durch meinem Musiksammlung und die Platte sofort ausgewählt. Es ist vielleicht diese jugendliche Energie die diese Musik ausstrahlt. Einem Gefühl welches ich seit dem ich 16 bin hinterher jage. Ach, wenigstens habe ich die Musik.</p> <p>Hier eine sehr seltene und schaubare Liveaufnahme der Band aus dem Jahr 1995.</p> <p><span class="videobox"> <iframe src="//player.vimeo.com/video/19126641?title=0&amp;byline=0&amp;portrait=0" width="640" height="390" frameborder="0" webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p>(<a href="http://www.jadetree.com/features2/2016/1/18/interview-with-victor-villarreal">via</a>)</p>Die Erdskulptur2016-03-10T15:33:00+01:002016-03-10T15:33:00+01:00marvintag:xsteadfastx.org,2016-03-10:/2016/03/10/die-erdskulptur/<p>Alexander Gerst hat mal gesagt, dass man am besten alle Menschen die auf der Welt Entscheidungen treffen, mal ins All senden sollte. Der Anblick der dünnen Athmosphere und der Erde als Ganzes, würde das Verständnis der Menschheit, im Platz des Universums, grundlegend verändern. Leider ist dies technisch utopisch. Diesen Anblick …</p><p>Alexander Gerst hat mal gesagt, dass man am besten alle Menschen die auf der Welt Entscheidungen treffen, mal ins All senden sollte. Der Anblick der dünnen Athmosphere und der Erde als Ganzes, würde das Verständnis der Menschheit, im Platz des Universums, grundlegend verändern. Leider ist dies technisch utopisch. Diesen Anblick am nächsten kommt man vielleicht bei der Ausstellung <a href="http://www.gasometer.de/de/ausstellungen/kommende-ausstellung/34-mehrsprachig/mehrsprachig-ausstellungen/264-wunder-der-natur">Wunder der Natur</a>. Dort wird die Erdoberfläche auf eine im Gasometer hängenden <a href="http://www.dlr.de/dlr/desktopdefault.aspx/tabid-10081/151_read-17109/#/gallery/22338">Kugel</a> projiziert. Und das in einer super hohen Auflösung.</p> <blockquote> <p>Ab dem 11. März 2016 können die Besucher des Gasometer Oberhausen einen ganz speziellen Blick auf den blauen Planeten werfen: Im mächtigen, 100 Meter hohen Turm des Industriedenkmals schwebt für die neue Ausstellung "Wunder der Erde" als Highlight eine Erdkugel mit 20 Metern Durchmessern. Aus Satellitendaten hat das Deutsche Zentrum für Luft- und Raumfahrt (DLR) Stück für Stück und Schicht für Schicht eine Animation erstellt, die von zwölf Projektoren auf die Erdkugel gespielt wird. Das Ergebnis: Eine Erdkugel, auf der sich Tag und Nacht abwechseln oder auch Wolkenbänder die Luftströmungen um die Erde sichtbar machen. Wer mit dem Panorama-Aufzug an der Innenwand des Gasometers in die Höhe fährt, blickt umgerechnet aus 36.000 Kilometern Entfernung auf den Erdball. Zudem zeigt der Gasometer Oberhausen 150 großformatige Fotos von Tieren und Pflanzen von namhaften Fotografen wie Frans Lanting, Tim Flach oder Anup Shah.</p> </blockquote> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/-abolRZQbzE' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Dokumentation über den Drum Doctor Ross Garfield2016-03-10T14:21:00+01:002016-03-10T14:21:00+01:00marvintag:xsteadfastx.org,2016-03-10:/2016/03/10/dokumentation-ueber-den-drum-doctor-ross-garfield/<p>Ross Garfield ist für sehr viele ikonische Drumsounds verantwortlich. Neben seiner <a href="http://www.drumdoctors.com/rental.htm">unfassbaren Sammlung</a>, ist seine Art zu stimmen berühmt und einzigartig. Wie sehr ich oft verzweifel am stimmen meines Drumsets. Man stelle sich vor seinen Beruf zu haben. Unfassbar.</p> <blockquote> <p>From Frank Sinatra to Johnny Cash, Michael Jackson to Metallica, master …</p></blockquote><p>Ross Garfield ist für sehr viele ikonische Drumsounds verantwortlich. Neben seiner <a href="http://www.drumdoctors.com/rental.htm">unfassbaren Sammlung</a>, ist seine Art zu stimmen berühmt und einzigartig. Wie sehr ich oft verzweifel am stimmen meines Drumsets. Man stelle sich vor seinen Beruf zu haben. Unfassbar.</p> <blockquote> <p>From Frank Sinatra to Johnny Cash, Michael Jackson to Metallica, master drum craftsman and technician, Ross Garfield, is responsible for some of the world’s most iconic drum sounds. This is the story of the man behind the drums.</p> </blockquote> <p><span class="videobox"> <iframe src="//player.vimeo.com/video/156509577?title=0&amp;byline=0&amp;portrait=0" width="640" height="390" frameborder="0" webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p>(<a href="https://www.reddit.com/r/drums/comments/49ojy5/the_drum_doctor/">via</a>)</p>CAFFEINATION: Ein Kaffeeroadtrip2016-03-08T14:52:00+01:002016-03-08T14:52:00+01:00marvintag:xsteadfastx.org,2016-03-08:/2016/03/08/caffeination-ein-kaffeeroadtrip/<p>Ich bin gerade durch Zufall auf eine ziemlich nette Videoserie über einen Kaffeeroadtrip durch American gestossen. Es wird in den jeweiligen Städten Kaffeeläden und Röstereien besucht. Wenn ich in fremden Städten unterwegs bin, sind Kaffees und Röstereien immer ein guter Weg sich eine Route durch die unererforschte Stadt zu bahnen …</p><p>Ich bin gerade durch Zufall auf eine ziemlich nette Videoserie über einen Kaffeeroadtrip durch American gestossen. Es wird in den jeweiligen Städten Kaffeeläden und Röstereien besucht. Wenn ich in fremden Städten unterwegs bin, sind Kaffees und Röstereien immer ein guter Weg sich eine Route durch die unererforschte Stadt zu bahnen.</p> <blockquote> <p>What makes a perfect cup of coffee? It starts with the bean, but many would say "its all about the Roast".</p> <p>The Pancake Epidemic presents it's first original web series, CAFFEINATION, starring their very own coffee specialist, Brandon Davenport; directed &amp; edited by video-maker, Jesse Meeker.</p> <p>This overly caffeinated road-trip brings Brandon around the Western and Central United States to discuss the roasting process and current coffee culture with dozens of different roasters.</p> <p>Stay tuned to all seven episodes to see if Brandon can reach his goal of drinking 100 cups of coffee during his road-trip through the Central and Western United States.</p> </blockquote> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/bPdFrT3fH-E' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/RHr2PDTtn18' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/HA7cowTmjWw' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/-sJJwtuxoDw' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/wUMDhjf-W_0' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/kfm8rkY-WPY' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/66mQC9Ng2zs' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/mpHEiuOK2Lg' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>NGINX im Container und docker-gen2016-03-01T09:48:00+01:002016-03-01T09:48:00+01:00marvintag:xsteadfastx.org,2016-03-01:/2016/03/01/nginx-im-container-und-docker-gen/<p><a href="http://giphy.com/gifs/fail-lol-achBohanYCPPG"><img src="http://media4.giphy.com/media/achBohanYCPPG/giphy.gif" alt="source: http://gifsboom.net/post/106865490574/fail-container-video"></a></p> <p>Das hatte ich lange auf meiner Liste. Bei mir lief immer noch der NGINX nicht im Container. Um ehrlich zu sein hatte ich mir alles schwieriger vorgestellt als es dann am Ende war. Ich halte mir immer alles in <code>docker-compose.yml</code>-Files fest.</p> <div class="highlight"><pre><span></span><span class="n">nginx</span><span class="o">:</span> <span class="n">image</span><span class="o">:</span> <span class="n">nginx</span> <span class="n">ports</span><span class="o">:</span> <span class="o">-</span> <span class="s2">&quot;80:80&quot;</span> <span class="o">-</span> <span class="s2">&quot;443 …</span></pre></div><p><a href="http://giphy.com/gifs/fail-lol-achBohanYCPPG"><img src="http://media4.giphy.com/media/achBohanYCPPG/giphy.gif" alt="source: http://gifsboom.net/post/106865490574/fail-container-video"></a></p> <p>Das hatte ich lange auf meiner Liste. Bei mir lief immer noch der NGINX nicht im Container. Um ehrlich zu sein hatte ich mir alles schwieriger vorgestellt als es dann am Ende war. Ich halte mir immer alles in <code>docker-compose.yml</code>-Files fest.</p> <div class="highlight"><pre><span></span><span class="n">nginx</span><span class="o">:</span> <span class="n">image</span><span class="o">:</span> <span class="n">nginx</span> <span class="n">ports</span><span class="o">:</span> <span class="o">-</span> <span class="s2">&quot;80:80&quot;</span> <span class="o">-</span> <span class="s2">&quot;443:443&quot;</span> <span class="n">volumes</span><span class="o">:</span> <span class="o">-</span> <span class="sr">/etc/nginx:/etc/</span><span class="n">nginx</span> <span class="o">-</span> <span class="sr">/etc/letsencrypt:/etc/</span><span class="n">letsencrypt</span> <span class="o">-</span> <span class="sr">/var/log/nginx:/var/log/</span><span class="n">nginx</span> </pre></div> <p>Dies läßt eine funktionierende Ubuntu-NGINX-Config mit dem <a href="https://hub.docker.com/_/nginx/">offiziellen NGINX-Dockerimage</a> laufen. Der NGINX macht fungiert nur als Reverse-Proxy für ein paar Anwendungen. Also werden Requests an ein Upstream Server weitergeleitet. NGINX macht manchmal Probleme wenn der Upstream Server Hostname nicht auflösbar ist und bricht dann sofort ab. Und dann sitzt man da und wundert sich wieso der Container immer wieder aussteigt. IP-Adressen sind auch auf ihre Art problematisch. Sie können sich unter Docker öfters ändern wenn man nach einem Image-Update den Container neustartet. Also was tun? Ich setze nun <a href="https://github.com/jwilder/docker-gen">docker-gen</a> ein. Der überwacht den Docker-Daemon und erstellt aus einem Template Config-Files. In diesem Fall eine <code>upstream.conf</code>. Die <code>docker-compose.yml</code> sieht so aus:</p> <div class="highlight"><pre><span></span><span class="n">gen</span><span class="o">:</span> <span class="n">image</span><span class="o">:</span> <span class="n">jwilder</span><span class="o">/</span><span class="n">docker</span><span class="o">-</span><span class="n">gen</span> <span class="n">volumes</span><span class="o">:</span> <span class="o">-</span> <span class="sr">/var/run/docker.sock:/tmp/</span><span class="n">docker</span><span class="o">.</span><span class="na">sock</span><span class="o">:</span><span class="n">ro</span> <span class="o">-</span> <span class="sr">/opt/docker-gen:/</span><span class="n">data</span> <span class="n">volumes_from</span><span class="o">:</span> <span class="o">-</span> <span class="n">nginx_nginx_1</span> <span class="n">command</span><span class="o">:</span> <span class="o">-</span><span class="n">notify</span><span class="o">-</span><span class="n">sighup</span> <span class="n">nginx_nginx_1</span> <span class="o">-</span><span class="n">watch</span> <span class="o">-</span><span class="n">only</span><span class="o">-</span><span class="n">exposed</span> <span class="sr">/data/nginx.tmpl /etc/nginx/</span><span class="n">upstream</span><span class="o">.</span><span class="na">conf</span> </pre></div> <p>docker-gen braucht den <code>docker.sock</code> um aus dem Container mit dem Daemon zu kommunizieren. Schließlich muss er mitbekommen wenn es Veränderungen gibt. Sprich Container gestartet werden oder Container nicht mehr da sind. Ich mounte das Arbeitsverzeichnis um auf das Template aus dem Container zugreifen zu können. Dazu binde ich alle Volumes des NGINX Containers ein. Schließlich soll das finale Config-File an die richtige Stelle geschoben werden. <code>-notify-sighup nginx_nginx_1</code> sagt das er den NGINX Container neustarten soll. Mit <code>-watch</code> beobachtet er den Docker Daemon auf Änderungen. Mit <code>-only-exposed</code> werden nur Container betrachtet die Ports exposen. Dahinter kommt dann das Template welches benutzt werden soll und wohin die Finale Datei geschrieben werden soll. Ich habe mich dafür entschieden nicht die ganze <code>nginx.conf</code> aus dem Template zu generieren. Ich lasse nur die <code>upstream.conf</code> bauen. Dieses inkludiere ich dann in der <code>nginx.conf</code>.</p> <div class="highlight"><pre><span></span><span class="nt">upstream</span> <span class="nt">cloud</span> <span class="p">{</span> <span class="err">#</span> <span class="err">ownclouddocker_owncloud_1</span> <span class="err">server</span> <span class="err">172.17.0.6:80</span><span class="p">;</span> <span class="p">}</span> <span class="nt">upstream</span> <span class="nt">gogs</span> <span class="p">{</span> <span class="err">#</span> <span class="err">gogs_gogs_1</span> <span class="err">server</span> <span class="err">172.17.0.5:3000</span><span class="p">;</span> <span class="p">}</span> <span class="nt">upstream</span> <span class="nt">ttrss</span> <span class="p">{</span> <span class="err">#</span> <span class="err">ttrssdocker_ttrss_1</span> <span class="err">server</span> <span class="err">172.17.0.8:80</span><span class="p">;</span> <span class="p">}</span> </pre></div> <p>Damit so eine Config erzeugt wird benutze ich dieses Template:</p> <div class="highlight"><pre><span></span><span class="cp">{{</span> <span class="nv">range</span> <span class="err">$</span><span class="nv">host</span><span class="o">,</span> <span class="err">$</span><span class="nv">containers</span> <span class="o">:=</span> <span class="nv">groupByMulti</span> <span class="err">$</span> <span class="s2">&quot;Env.VIRTUAL_HOST&quot;</span> <span class="s2">&quot;,&quot;</span> <span class="cp">}}</span><span class="x"></span> <span class="x">upstream </span><span class="cp">{{</span> <span class="err">$</span><span class="nv">host</span> <span class="cp">}}</span><span class="x"> {</span> <span class="cp">{{</span> <span class="nv">range</span> <span class="err">$</span><span class="nv">index</span><span class="o">,</span> <span class="err">$</span><span class="nv">value</span> <span class="o">:=</span> <span class="err">$</span><span class="nv">containers</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="err">$</span><span class="nv">addrLen</span> <span class="o">:=</span> <span class="nv">len</span> <span class="err">$</span><span class="nv">value.Addresses</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="err">$</span><span class="nv">network</span> <span class="o">:=</span> <span class="nv">index</span> <span class="err">$</span><span class="nv">value.Networks</span> <span class="m">0</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span><span class="o">/*</span> <span class="nv">If</span> <span class="nv">only</span> <span class="m">1</span> <span class="nv">port</span> <span class="nv">exposed</span><span class="o">,</span> <span class="nv">use</span> <span class="nv">that</span> <span class="o">*/</span><span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="k">if</span> <span class="nv">eq</span> <span class="err">$</span><span class="nv">addrLen</span> <span class="m">1</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="k">with</span> <span class="err">$</span><span class="nv">address</span> <span class="o">:=</span> <span class="nv">index</span> <span class="err">$</span><span class="nv">value.Addresses</span> <span class="m">0</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> # </span><span class="cp">{{</span><span class="err">$</span><span class="nv">value.Name</span><span class="cp">}}</span><span class="x"></span> <span class="x"> server </span><span class="cp">{{</span> <span class="err">$</span><span class="nv">network.IP</span> <span class="cp">}}</span><span class="x">:</span><span class="cp">{{</span> <span class="err">$</span><span class="nv">address.Port</span> <span class="cp">}}</span><span class="x">;</span> <span class="x"> </span><span class="cp">{{</span> <span class="nv">end</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span><span class="o">/*</span> <span class="nv">If</span> <span class="nv">more</span> <span class="nv">than</span> <span class="nv">one</span> <span class="nv">port</span> <span class="nv">exposed</span><span class="o">,</span> <span class="nv">use</span> <span class="nv">the</span> <span class="nv">one</span> <span class="nv">matching</span> <span class="nv">VIRTUAL_PORT</span> <span class="nv">env</span> <span class="nv">var</span> <span class="o">*/</span><span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="k">else</span> <span class="k">if</span> <span class="err">$</span><span class="nv">value.Env.VIRTUAL_PORT</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="nv">range</span> <span class="err">$</span><span class="nv">i</span><span class="o">,</span> <span class="err">$</span><span class="nv">address</span> <span class="o">:=</span> <span class="err">$</span><span class="nv">value.Addresses</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="k">if</span> <span class="nv">eq</span> <span class="err">$</span><span class="nv">address.Port</span> <span class="err">$</span><span class="nv">value.Env.VIRTUAL_PORT</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> # </span><span class="cp">{{</span><span class="err">$</span><span class="nv">value.Name</span><span class="cp">}}</span><span class="x"></span> <span class="x"> server </span><span class="cp">{{</span> <span class="err">$</span><span class="nv">network.IP</span> <span class="cp">}}</span><span class="x">:</span><span class="cp">{{</span> <span class="err">$</span><span class="nv">address.Port</span> <span class="cp">}}</span><span class="x">;</span> <span class="x"> </span><span class="cp">{{</span> <span class="nv">end</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="nv">end</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span><span class="o">/*</span> <span class="nv">Else</span> <span class="nv">default</span> <span class="nv">to</span> <span class="nv">standard</span> <span class="nv">web</span> <span class="nv">port</span> <span class="m">80</span> <span class="o">*/</span><span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="k">else</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="nv">range</span> <span class="err">$</span><span class="nv">i</span><span class="o">,</span> <span class="err">$</span><span class="nv">address</span> <span class="o">:=</span> <span class="err">$</span><span class="nv">value.Addresses</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="k">if</span> <span class="nv">eq</span> <span class="err">$</span><span class="nv">address.Port</span> <span class="s2">&quot;80&quot;</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> # </span><span class="cp">{{</span><span class="err">$</span><span class="nv">value.Name</span><span class="cp">}}</span><span class="x"></span> <span class="x"> server </span><span class="cp">{{</span> <span class="err">$</span><span class="nv">network.IP</span> <span class="cp">}}</span><span class="x">:</span><span class="cp">{{</span> <span class="err">$</span><span class="nv">address.Port</span> <span class="cp">}}</span><span class="x">;</span> <span class="x"> </span><span class="cp">{{</span> <span class="nv">end</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="nv">end</span> <span class="cp">}}</span><span class="x"></span> <span class="x"> </span><span class="cp">{{</span> <span class="nv">end</span> <span class="cp">}}</span><span class="x"></span> <span class="cp">{{</span> <span class="nv">end</span> <span class="cp">}}</span><span class="x"></span> <span class="x">}</span> <span class="cp">{{</span> <span class="nv">end</span> <span class="cp">}}</span><span class="x"></span> </pre></div> <p>Damit <code>docker-gen</code> die passenden Container findet die er betrachten soll, müssen den Containern ein paar Environment-Variabeln mitgegeben werden. Hier zum Beispiel mit <a href="https://gogs.io">Gogs</a>:</p> <div class="highlight"><pre><span></span><span class="n">gogs</span><span class="o">:</span> <span class="n">image</span><span class="o">:</span> <span class="n">gogs</span><span class="o">/</span><span class="n">gogs</span> <span class="n">ports</span><span class="o">:</span> <span class="o">-</span> <span class="s2">&quot;3000&quot;</span> <span class="n">volumes</span><span class="o">:</span> <span class="o">-</span> <span class="s2">&quot;/srv/www/gogs:/data&quot;</span> <span class="n">environment</span><span class="o">:</span> <span class="o">-</span> <span class="n">VIRTUAL_HOST</span><span class="o">=</span><span class="n">gogs</span> <span class="o">-</span> <span class="n">VIRTUAL_PORT</span><span class="o">=</span><span class="mi">3000</span> </pre></div> <p><code>VIRTUAL_HOST</code> gibt dem ganzen einen Namen. Den brauchen wir dann in der eigentlichen vhost-Config. <code>VIRTUAL_PORT</code> wird gebraucht wenn der Port nicht <code>80</code> ist. Dann ignoriert <code>docker-gen</code> sonst den Container.</p> <p>Die NGINX-Config sieht dann so aus:</p> <div class="highlight"><pre><span></span><span class="nt">server</span> <span class="p">{</span> <span class="err">listen</span> <span class="err">443</span><span class="p">;</span> <span class="err">server_name</span> <span class="err">git.foo.bar</span><span class="p">;</span> <span class="err">ssl</span> <span class="err">on</span><span class="p">;</span> <span class="err">ssl_certificate</span> <span class="err">/etc/letsencrypt/live/git.foo.bar/fullchain.pem</span><span class="p">;</span> <span class="err">ssl_certificate_key</span> <span class="err">/etc/letsencrypt/live/git.foo.bar/privkey.pem</span><span class="p">;</span> <span class="err">client_max_body_size</span> <span class="err">50m</span><span class="p">;</span> <span class="err">location</span> <span class="err">/</span> <span class="err">{</span> <span class="err">proxy_pass</span> <span class="n">http</span><span class="p">:</span><span class="o">//</span><span class="n">gogs</span><span class="p">;</span> <span class="err">proxy_set_header</span> <span class="err">X-Forwarded-Host</span> <span class="err">$server_name</span><span class="p">;</span> <span class="err">proxy_set_header</span> <span class="err">X-Forwarded-Proto</span> <span class="err">https</span><span class="p">;</span> <span class="err">proxy_set_header</span> <span class="err">X-Forwarded-For</span> <span class="err">$remote_addr</span><span class="p">;</span> <span class="p">}</span> <span class="err">}</span> </pre></div> <p>Unter <code>proxy_pass</code> taucht <code>VIRTUAL_HOST</code> wieder auf. In der NGINX Config spiegelt sich das wie folgt wieder:</p> <div class="highlight"><pre><span></span>http { ## # Basic Settings ## ... ... ... ## # Virtual Host Configs ## include /etc/nginx/conf.d/*.conf; include /etc/nginx/sites-enabled/*; ## # Upstream ## include /etc/nginx/upstream.conf; } </pre></div> <p>Nun alles Container starten und genießen oder sowas.</p>Der Tag an dem Umberto Eco starb2016-02-29T09:18:00+01:002016-02-29T09:18:00+01:00marvintag:xsteadfastx.org,2016-02-29:/2016/02/29/der-tag-an-dem-umberto-eco-starb/<p><a href="https://www.flickr.com/photos/marvinxsteadfast/7920500522/"><img src="https://farm9.staticflickr.com/8310/7920500522_3f6598f37c_b.jpg" alt="Serravalle Pistoiese"></a></p> <p>Eigentlich bin ich kein großer Hörbuch-Fan. Doch seit ein paar Minaten habe ich ein Audible-Abo. Ich fahre jeden Tag mit dem Fahrrad ins Büro und wenn die Podcasts leer gehört sind, greife ich auf das ein oder andere Hörbuch zurück. Ich habe gemerkt das es mir oft ziemlich schwer fällt …</p><p><a href="https://www.flickr.com/photos/marvinxsteadfast/7920500522/"><img src="https://farm9.staticflickr.com/8310/7920500522_3f6598f37c_b.jpg" alt="Serravalle Pistoiese"></a></p> <p>Eigentlich bin ich kein großer Hörbuch-Fan. Doch seit ein paar Minaten habe ich ein Audible-Abo. Ich fahre jeden Tag mit dem Fahrrad ins Büro und wenn die Podcasts leer gehört sind, greife ich auf das ein oder andere Hörbuch zurück. Ich habe gemerkt das es mir oft ziemlich schwer fällt mich auf die aufgezeichneten Sätze zu konzentrieren. Ich weiß das ich ein sehr langsamer Leser bin und so fehlt mir mein eigener Rhytmus und vielleicht auch das Gefühl von Papier zwischen meinen Fingern. Als ich also Donnerstag das neue Audible-Guthaben erhiert, wollte ich es in eins meiner Lieblingsbücher investieren. Diese Art von Bücher die man immer wieder gern liest. Etwas was ich schon kannte und meinen monotonen Arbeitsweg ein wenig aufhellen kann. Nicht etwa durch seine fröhlichkeit, sondern durch die wunderschänen Sätze. Ich musste nicht lange überlegen. Die Wahl viel sofort auf <a href="https://de.wikipedia.org/wiki/Der_Name_der_Rose">Der Name der Rose</a>. Das Buch las ich ich zum ersten mal vor ein paar Jahren in der Toskana. Dies war mehr ein Zufalls als eine geplante, örtlich passende, Urlaubslektüre. Als ich vor dem alten toskanischen Bauernhaus saß und das Buch zum ersten mal aufschlug,und nach ein paar Seiten merkte wo die Geschichte spielt. Ich malte mir aus, das die Abtei sei identisch mit den alten Ruinen die wir am Tag vorher auf einen Hügel entdeckt hatten. Im morgendlichen toskanischen Nebel stellte ich mir vor der Novize Adson von Merl zu sein und flog durch die Seiten des Buches. Dies war der Beginn meiner Faszination für Umberto Eco. Ich habe bei welten nicht alles gelesen oder gar verstanden was er veröffentlich hat. Aber seine Wissenswurt und sein nicht endendes Interesse in allen Bereichen der Wissenschaft und Kultur sind eine riesige Inspiration. Seine private Bibliothek und seine These zur <a href="https://www.brainpickings.org/2015/03/24/umberto-eco-antilibrary/">Anti-Bibliothek</a> sind Teil meiner täglichen Träumerein. Und nicht zuletzt haben wir unseren Kate nach ihm benannt. Diese Entscheidung fiel sehr schnell. Am Morgen des Samstags, nachdem ich die ersten Kapitel "Der Name der Rose" gehört habe, las ich von seinem Tod. In einem Jahr wo fast wöchentlich eine Legende stirbt, ist es doch der Tod Ecos der ein mich am meisten berührt.</p>Zwei Videos über Sealand2016-02-19T14:20:00+01:002016-02-19T14:20:00+01:00marvintag:xsteadfastx.org,2016-02-19:/2016/02/19/zwei-videos-ueber-sealand/<p><a href="https://de.wikipedia.org/wiki/Sealand">Sealand</a> ist immer ontopic. Die nicht offiziell anerkannte Mikronation vor England. Steng genommen ist Sealand nur eine alte Militärbasis auf dem offenden Meer, eingenommen von einem gewissen Paddy Roy Bates. Die Geschichte von Sealand bietet einige Highlights. Eine versuchte Eroberung des Landes durch Dritte, inklusive Geiselnahme, und den Traum eines …</p><p><a href="https://de.wikipedia.org/wiki/Sealand">Sealand</a> ist immer ontopic. Die nicht offiziell anerkannte Mikronation vor England. Steng genommen ist Sealand nur eine alte Militärbasis auf dem offenden Meer, eingenommen von einem gewissen Paddy Roy Bates. Die Geschichte von Sealand bietet einige Highlights. Eine versuchte Eroberung des Landes durch Dritte, inklusive Geiselnahme, und den Traum eines sicheren <a href="https://de.wikipedia.org/wiki/Datenhafen">Datenhafens</a>.</p> <p>Hier ein kleines animiertes Video über die Geschichte von Sealand.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/99uvmq2P7kU' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p>Einer der Admins die den Datenhafen betriebenhat auf dem letzten Chaos Communication Congress einen Vortrag gehalten. Ziemlich interessant.</p> <iframe width="853" height="480" src="https://media.ccc.de/v/32c3-7432-datahavens_from_havenco_to_today/oembed" frameborder="0" allowfullscreen></iframe>Viva Dada: ARTE Dokumentation2016-02-19T12:11:00+01:002016-02-19T12:11:00+01:00marvintag:xsteadfastx.org,2016-02-19:/2016/02/19/viva-dada-arte-dokumentation/<p>Arte hat mal wieder eine tolle <a href="http://www.arte.tv/guide/de/058936-000-A/viva-dada?vid=058936-000-A_CLIP_EXTRAIT_DE_de">Dokumentation</a> produziert. Es geht um Dada. Anläßlich zum 100 jährigen Geburtstag.</p> <blockquote> <p>Die Kunst ist tot, es lebe Dada! Vor 100 Jahren, im Februar 1916, rief Hugo Ball mit dem Manifest zum ersten Dada-Abend in Zürich eine neue Kunstrichtung aus. Der Dadaismus formierte sich im …</p></blockquote><p>Arte hat mal wieder eine tolle <a href="http://www.arte.tv/guide/de/058936-000-A/viva-dada?vid=058936-000-A_CLIP_EXTRAIT_DE_de">Dokumentation</a> produziert. Es geht um Dada. Anläßlich zum 100 jährigen Geburtstag.</p> <blockquote> <p>Die Kunst ist tot, es lebe Dada! Vor 100 Jahren, im Februar 1916, rief Hugo Ball mit dem Manifest zum ersten Dada-Abend in Zürich eine neue Kunstrichtung aus. Der Dadaismus formierte sich im Ersten Weltkrieg, als Schrei des Aufbegehrens gegen eine Gesellschaft, die solche Kriegsgräuel zu verantworten hatte. Der Dadaismus wurde bald zu einer internationalen Avantgarde-Bewegung.</p> </blockquote> <iframe src="http://www.arte.tv/guide/de/embed/058936-000-A/medium" allowfullscreen="true" style="width: 600px; height: 344px;" frameborder="0"></iframe>Ansible Playbooks und variable Hosts2016-02-18T10:23:00+01:002016-02-18T10:23:00+01:00marvintag:xsteadfastx.org,2016-02-18:/2016/02/18/ansible-playbooks-und-variable-hosts/<p>Ich benutze <a href="http://ansible.com/">Ansible</a> für viele Sachen. Unter anderem habe ich ein <a href="https://github.com/xsteadfastx/batcave">Set an Playbooks und Roles</a> um meine Server so anzupassen das ich mich auf ihnen wohl fühle. Bis jetzt habe ich immer Ansible auch auf den Servern installiert und es dann lokal ausgeführt. Aber dabei nutze ich nicht das …</p><p>Ich benutze <a href="http://ansible.com/">Ansible</a> für viele Sachen. Unter anderem habe ich ein <a href="https://github.com/xsteadfastx/batcave">Set an Playbooks und Roles</a> um meine Server so anzupassen das ich mich auf ihnen wohl fühle. Bis jetzt habe ich immer Ansible auch auf den Servern installiert und es dann lokal ausgeführt. Aber dabei nutze ich nicht das eigentlich Ansible Feature: das Ganze Remote über SSH auszurollen. Problem: In den Playbooks muss es die Zeile <code>- hosts:</code> geben. Aber eigentlich will ich das alles Variabel ausführen können. Zum Beispiel nutze ich die gleichen Files auch um meine Vagrant Container einzurichten. Wieso also beim Aufruf die Hosts dem Playbook nicht einfach übergeben? Die <a href="https://github.com/k4ml/importerror/blob/master/posts/ansible-playbook-specify-hosts-on-the-command-line.md">Lösung</a> ist dann doch wieder einmal einfacher als man denkt. Man benutzt die Möglichkeit in Playbooks und Roles Variabeln einzusetzen.</p> <div class="highlight"><pre><span></span><span class="x">---</span> <span class="x">- hosts: &quot;</span><span class="cp">{{</span> <span class="nv">hosts</span> <span class="cp">}}</span><span class="x">&quot;</span> <span class="x"> roles:</span> <span class="x"> - git</span> <span class="x"> - tmux</span> <span class="x"> - vim</span> <span class="x"> - zsh</span> </pre></div> <p>Diese Variabel übergeben wir beim Aufruf von Ansible.</p> <p><code>ansible-playbook base.yml --extra-vars="hosts=myservers"</code></p>WDR 3 Kulturfeature über James Turrell2016-02-16T12:04:00+01:002016-02-16T12:04:00+01:00marvintag:xsteadfastx.org,2016-02-16:/2016/02/16/wdr-3-kulturfeature-ueber-james-turrell/<p><a href="https://commons.wikimedia.org/wiki/File:Roden.jpg"><img alt="Roden Crater" src="http://xsteadfastx.org/images/roden_crater.jpg"></a></p> <p>Das Wolfsburg Project von James Turrell hat bei mir eine bleibenden Eindruck hinterlassen. Weiß ich doch zimelich wenig von Kunst und lasse sie einfach auf mich wirken. Wie verändert sie die Atmosphere für mich, wie wirkt sie auf den Raum. James Turrell hat dieses betretbare, riesige Werk geschaffen. Stand ich …</p><p><a href="https://commons.wikimedia.org/wiki/File:Roden.jpg"><img alt="Roden Crater" src="http://xsteadfastx.org/images/roden_crater.jpg"></a></p> <p>Das Wolfsburg Project von James Turrell hat bei mir eine bleibenden Eindruck hinterlassen. Weiß ich doch zimelich wenig von Kunst und lasse sie einfach auf mich wirken. Wie verändert sie die Atmosphere für mich, wie wirkt sie auf den Raum. James Turrell hat dieses betretbare, riesige Werk geschaffen. Stand ich drin, zog es mir fast die Schuhe aus. Das Licht und die nicht mehr warnehmbaren Wände verwandelten meine Beine in Pudding. In einer zusätzlichen Ausstellung ging es um die anderen Werke James Turrells. Dort erfuhr ich das erste mal von seinem <a href="https://de.wikipedia.org/wiki/Roden_Crater">Roden Crater</a>. Einen Vulkan den er in den 70ern entdeckte als er ein Atelier aus dem Flugzeug suchte. Diesen baut er seit 1974 zu einer Kathedrale für seine Kunst aus. Immer noch nicht für die Öffentlichkeit eröffnet. Ein Lebenswerk und vielleicht das größte Werk der zeitgenössischen Kunst.</p> <p>Der WDR hat ein <a href="http://www.wdr3.de/programm/sendungen/wdr3kulturfeature/open-sky-james-turrell-100.html">Feature</a> über James Turrell und den Roden Crater veröffentlicht. Absolute Hörempfehlung.</p> <blockquote> <p>WDR 3 Kulturfeature: Ein Stipendium der Guggenheim Stiftung investierte der Lichtkünstler James Turrell 1974 in Kerosin und flog mit seinem Flugzeug 700 Stunden über die Weite des amerikanischen Westens, bis er Roden Crater entdeckte. Autor: Peter-Moritz Pickshaus; Redaktion: Leslie Rosin; Produktion: NDR/WDR 2012 © WDR 2016</p> </blockquote> <p><audio controls><source src="http://podcast-ww.wdr.de/medstdp/fsk0/92/929938/wdr3kulturfeature_2016-02-13_jamesturrellserloschenervulkanrodencrateropenskywdr3kulturfeature13022016_wdr3.mp3" type="audio/mpeg">Your browser does not support the audio element.</audio></p> <p>Das Wolfsburg Project:</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/QWekIcZaKns' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p>Hier noch eine Dokumentation über den Krater:</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/nPI65cEXbXI' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/YtsgUu7dO5k' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Nochmal Serien Plots, XKCD Style2016-02-12T14:22:00+01:002016-02-12T14:22:00+01:00marvintag:xsteadfastx.org,2016-02-12:/2016/02/12/nochmal-serien-plots-xkcd-style/<p> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Immer noch ziemlich nutzlos aber diesmal mit XKCD-Eyecandy. <a href="http://bokeh.pydata.org/">Bokeh</a> ist ja ganz schön, braucht aber eine ziemlich große JS-Komponente die immer geladen werden muss. Mit <a href="http://matplotlib.org/">matplotlib</a> kann man statische Grafen zeichnen und mit <code>%matplotlib notebook</code> vorher einen Auschnitt wählen.</p> <p>Ich wollte mal schauen wie sich die Ratingverläufe der Staffeln einer …</p></div></div></div></p><p> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Immer noch ziemlich nutzlos aber diesmal mit XKCD-Eyecandy. <a href="http://bokeh.pydata.org/">Bokeh</a> ist ja ganz schön, braucht aber eine ziemlich große JS-Komponente die immer geladen werden muss. Mit <a href="http://matplotlib.org/">matplotlib</a> kann man statische Grafen zeichnen und mit <code>%matplotlib notebook</code> vorher einen Auschnitt wählen.</p> <p>Ich wollte mal schauen wie sich die Ratingverläufe der Staffeln einer Serie entfalten. Eventuell kann man ja ein daramaturgischen Verlauf erkennen. Ich als jemand der selber keine Ratings abgibt. Alles befindet sich doch ziemlich auf einem Level.</p> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[1]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="kn">import</span> <span class="nn">pandas</span> <span class="kn">as</span> <span class="nn">pd</span> <span class="kn">import</span> <span class="nn">matplotlib</span> <span class="kn">import</span> <span class="nn">numpy</span> <span class="kn">as</span> <span class="nn">np</span> <span class="kn">from</span> <span class="nn">trakt.tv</span> <span class="kn">import</span> <span class="n">TVShow</span> </pre></div> </div> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Nach dem import der Libs führen wir folgendes Jupyter-Magic aus um die erzeugten Plots im Notebook noch leicht anpassen zu könne.</p> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[2]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="o">%</span><span class="k">matplotlib</span> <span class="n">notebook</span> </pre></div> </div> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Wir aktivieren die XKCD-Style-Plots:</p> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[3]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="n">matplotlib</span><span class="o">.</span><span class="n">pyplot</span><span class="o">.</span><span class="n">xkcd</span><span class="p">()</span> </pre></div> </div> </div> </div> <div class="output_wrapper"> <div class="output"> <div class="output_area"><div class="prompt output_prompt">Out[3]:</div> <div class="output_text output_subarea output_pyout"> <pre> &lt;matplotlib.rc_context at 0xaf0ef58c&gt; </pre> </div> </div> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Nun kommt die Funktion die alle Daten einsammelt und ausgibt:</p> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[4]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="k">def</span> <span class="nf">season_ratings</span><span class="p">(</span><span class="n">name</span><span class="p">):</span> <span class="n">tv_show</span> <span class="o">=</span> <span class="n">TVShow</span><span class="p">(</span><span class="n">name</span><span class="p">)</span> <span class="n">data</span> <span class="o">=</span> <span class="p">{}</span> <span class="k">for</span> <span class="n">season</span> <span class="ow">in</span> <span class="n">tv_show</span><span class="o">.</span><span class="n">seasons</span><span class="p">:</span> <span class="k">if</span> <span class="n">season</span><span class="o">.</span><span class="n">season</span> <span class="o">==</span> <span class="mi">0</span><span class="p">:</span> <span class="k">continue</span> <span class="n">ratings</span> <span class="o">=</span> <span class="p">[]</span> <span class="k">for</span> <span class="n">episode</span> <span class="ow">in</span> <span class="n">season</span><span class="o">.</span><span class="n">episodes</span><span class="p">:</span> <span class="n">ratings</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="n">episode</span><span class="o">.</span><span class="n">rating</span><span class="p">)</span> <span class="n">data</span><span class="p">[</span><span class="n">season</span><span class="o">.</span><span class="n">season</span><span class="p">]</span> <span class="o">=</span> <span class="n">pd</span><span class="o">.</span><span class="n">Series</span><span class="p">(</span><span class="n">ratings</span><span class="p">)</span> <span class="n">df</span> <span class="o">=</span> <span class="n">pd</span><span class="o">.</span><span class="n">DataFrame</span><span class="p">(</span><span class="n">data</span><span class="p">)</span> <span class="n">df</span><span class="o">.</span><span class="n">plot</span><span class="p">(</span><span class="n">colormap</span><span class="o">=</span><span class="n">matplotlib</span><span class="o">.</span><span class="n">cm</span><span class="o">.</span><span class="n">Accent</span><span class="p">)</span> </pre></div> </div> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <h2 id="Beispiele">Beispiele<a class="anchor-link" href="#Beispiele">&#182;</a></h2> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[5]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="n">season_ratings</span><span class="p">(</span><span class="s1">&#39;The IT-Crowd&#39;</span><span class="p">)</span> </pre></div> </div> </div> </div> <div class="output_wrapper"> <div class="output"> <div class="output_area"><div class="prompt"></div> <div class="output_subarea output_javascript "> <script type="text/javascript"> /* Put everything inside the global mpl namespace */ window.mpl = {}; mpl.get_websocket_type = function() { if (typeof(WebSocket) !== 'undefined') { return WebSocket; } else if (typeof(MozWebSocket) !== 'undefined') { return MozWebSocket; } else { alert('Your browser does not have WebSocket support.' + 'Please try Chrome, Safari or Firefox ≥ 6. ' + 'Firefox 4 and 5 are also supported but you ' + 'have to enable WebSockets in about:config.'); }; } mpl.figure = function(figure_id, websocket, ondownload, parent_element) { this.id = figure_id; this.ws = websocket; this.supports_binary = (this.ws.binaryType != undefined); if (!this.supports_binary) { var warnings = document.getElementById("mpl-warnings"); if (warnings) { warnings.style.display = 'block'; warnings.textContent = ( "This browser does not support binary websocket messages. " + "Performance may be slow."); } } this.imageObj = new Image(); this.context = undefined; this.message = undefined; this.canvas = undefined; this.rubberband_canvas = undefined; this.rubberband_context = undefined; this.format_dropdown = undefined; this.image_mode = 'full'; this.root = $('<div/>'); this._root_extra_style(this.root) this.root.attr('style', 'display: inline-block'); $(parent_element).append(this.root); this._init_header(this); this._init_canvas(this); this._init_toolbar(this); var fig = this; this.waiting = false; this.ws.onopen = function () { fig.send_message("supports_binary", {value: fig.supports_binary}); fig.send_message("send_image_mode", {}); fig.send_message("refresh", {}); } this.imageObj.onload = function() { if (fig.image_mode == 'full') { // Full images could contain transparency (where diff images // almost always do), so we need to clear the canvas so that // there is no ghosting. fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height); } fig.context.drawImage(fig.imageObj, 0, 0); }; this.imageObj.onunload = function() { this.ws.close(); } this.ws.onmessage = this._make_on_message_function(this); this.ondownload = ondownload; } mpl.figure.prototype._init_header = function() { var titlebar = $( '<div class="ui-dialog-titlebar ui-widget-header ui-corner-all ' + 'ui-helper-clearfix"/>'); var titletext = $( '<div class="ui-dialog-title" style="width: 100%; ' + 'text-align: center; padding: 3px;"/>'); titlebar.append(titletext) this.root.append(titlebar); this.header = titletext[0]; } mpl.figure.prototype._canvas_extra_style = function(canvas_div) { } mpl.figure.prototype._root_extra_style = function(canvas_div) { } mpl.figure.prototype._init_canvas = function() { var fig = this; var canvas_div = $('<div/>'); canvas_div.attr('style', 'position: relative; clear: both; outline: 0'); function canvas_keyboard_event(event) { return fig.key_event(event, event['data']); } canvas_div.keydown('key_press', canvas_keyboard_event); canvas_div.keyup('key_release', canvas_keyboard_event); this.canvas_div = canvas_div this._canvas_extra_style(canvas_div) this.root.append(canvas_div); var canvas = $('<canvas/>'); canvas.addClass('mpl-canvas'); canvas.attr('style', "left: 0; top: 0; z-index: 0; outline: 0") this.canvas = canvas[0]; this.context = canvas[0].getContext("2d"); var rubberband = $('<canvas/>'); rubberband.attr('style', "position: absolute; left: 0; top: 0; z-index: 1;") var pass_mouse_events = true; canvas_div.resizable({ start: function(event, ui) { pass_mouse_events = false; }, resize: function(event, ui) { fig.request_resize(ui.size.width, ui.size.height); }, stop: function(event, ui) { pass_mouse_events = true; fig.request_resize(ui.size.width, ui.size.height); }, }); function mouse_event_fn(event) { if (pass_mouse_events) return fig.mouse_event(event, event['data']); } rubberband.mousedown('button_press', mouse_event_fn); rubberband.mouseup('button_release', mouse_event_fn); // Throttle sequential mouse events to 1 every 20ms. rubberband.mousemove('motion_notify', mouse_event_fn); rubberband.mouseenter('figure_enter', mouse_event_fn); rubberband.mouseleave('figure_leave', mouse_event_fn); canvas_div.on("wheel", function (event) { event = event.originalEvent; event['data'] = 'scroll' if (event.deltaY < 0) { event.step = 1; } else { event.step = -1; } mouse_event_fn(event); }); canvas_div.append(canvas); canvas_div.append(rubberband); this.rubberband = rubberband; this.rubberband_canvas = rubberband[0]; this.rubberband_context = rubberband[0].getContext("2d"); this.rubberband_context.strokeStyle = "#000000"; this._resize_canvas = function(width, height) { // Keep the size of the canvas, canvas container, and rubber band // canvas in synch. canvas_div.css('width', width) canvas_div.css('height', height) canvas.attr('width', width); canvas.attr('height', height); rubberband.attr('width', width); rubberband.attr('height', height); } // Set the figure to an initial 600x600px, this will subsequently be updated // upon first draw. this._resize_canvas(600, 600); // Disable right mouse context menu. $(this.rubberband_canvas).bind("contextmenu",function(e){ return false; }); function set_focus () { canvas.focus(); canvas_div.focus(); } window.setTimeout(set_focus, 100); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items) { var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { // put a spacer in here. continue; } var button = $('<button/>'); button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' + 'ui-button-icon-only'); button.attr('role', 'button'); button.attr('aria-disabled', 'false'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); var icon_img = $('<span/>'); icon_img.addClass('ui-button-icon-primary ui-icon'); icon_img.addClass(image); icon_img.addClass('ui-corner-all'); var tooltip_span = $('<span/>'); tooltip_span.addClass('ui-button-text'); tooltip_span.html(tooltip); button.append(icon_img); button.append(tooltip_span); nav_element.append(button); } var fmt_picker_span = $('<span/>'); var fmt_picker = $('<select/>'); fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content'); fmt_picker_span.append(fmt_picker); nav_element.append(fmt_picker_span); this.format_dropdown = fmt_picker[0]; for (var ind in mpl.extensions) { var fmt = mpl.extensions[ind]; var option = $( '<option/>', {selected: fmt === mpl.default_extension}).html(fmt); fmt_picker.append(option) } // Add hover states to the ui-buttons $( ".ui-button" ).hover( function() { $(this).addClass("ui-state-hover");}, function() { $(this).removeClass("ui-state-hover");} ); var status_bar = $('<span class="mpl-message"/>'); nav_element.append(status_bar); this.message = status_bar[0]; } mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) { // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client, // which will in turn request a refresh of the image. this.send_message('resize', {'width': x_pixels, 'height': y_pixels}); } mpl.figure.prototype.send_message = function(type, properties) { properties['type'] = type; properties['figure_id'] = this.id; this.ws.send(JSON.stringify(properties)); } mpl.figure.prototype.send_draw_message = function() { if (!this.waiting) { this.waiting = true; this.ws.send(JSON.stringify({type: "draw", figure_id: this.id})); } } mpl.figure.prototype.handle_save = function(fig, msg) { var format_dropdown = fig.format_dropdown; var format = format_dropdown.options[format_dropdown.selectedIndex].value; fig.ondownload(fig, format); } mpl.figure.prototype.handle_resize = function(fig, msg) { var size = msg['size']; if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) { fig._resize_canvas(size[0], size[1]); fig.send_message("refresh", {}); }; } mpl.figure.prototype.handle_rubberband = function(fig, msg) { var x0 = msg['x0']; var y0 = fig.canvas.height - msg['y0']; var x1 = msg['x1']; var y1 = fig.canvas.height - msg['y1']; x0 = Math.floor(x0) + 0.5; y0 = Math.floor(y0) + 0.5; x1 = Math.floor(x1) + 0.5; y1 = Math.floor(y1) + 0.5; var min_x = Math.min(x0, x1); var min_y = Math.min(y0, y1); var width = Math.abs(x1 - x0); var height = Math.abs(y1 - y0); fig.rubberband_context.clearRect( 0, 0, fig.canvas.width, fig.canvas.height); fig.rubberband_context.strokeRect(min_x, min_y, width, height); } mpl.figure.prototype.handle_figure_label = function(fig, msg) { // Updates the figure title. fig.header.textContent = msg['label']; } mpl.figure.prototype.handle_cursor = function(fig, msg) { var cursor = msg['cursor']; switch(cursor) { case 0: cursor = 'pointer'; break; case 1: cursor = 'default'; break; case 2: cursor = 'crosshair'; break; case 3: cursor = 'move'; break; } fig.rubberband_canvas.style.cursor = cursor; } mpl.figure.prototype.handle_message = function(fig, msg) { fig.message.textContent = msg['message']; } mpl.figure.prototype.handle_draw = function(fig, msg) { // Request the server to send over a new figure. fig.send_draw_message(); } mpl.figure.prototype.handle_image_mode = function(fig, msg) { fig.image_mode = msg['mode']; } mpl.figure.prototype.updated_canvas_event = function() { // Called whenever the canvas gets updated. this.send_message("ack", {}); } // A function to construct a web socket function for onmessage handling. // Called in the figure constructor. mpl.figure.prototype._make_on_message_function = function(fig) { return function socket_on_message(evt) { if (evt.data instanceof Blob) { /* FIXME: We get "Resource interpreted as Image but * transferred with MIME type text/plain:" errors on * Chrome. But how to set the MIME type? It doesn't seem * to be part of the websocket stream */ evt.data.type = "image/png"; /* Free the memory for the previous frames */ if (fig.imageObj.src) { (window.URL || window.webkitURL).revokeObjectURL( fig.imageObj.src); } fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL( evt.data); fig.updated_canvas_event(); fig.waiting = false; return; } else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == "data:image/png;base64") { fig.imageObj.src = evt.data; fig.updated_canvas_event(); fig.waiting = false; return; } var msg = JSON.parse(evt.data); var msg_type = msg['type']; // Call the "handle_{type}" callback, which takes // the figure and JSON message as its only arguments. try { var callback = fig["handle_" + msg_type]; } catch (e) { console.log("No handler for the '" + msg_type + "' message type: ", msg); return; } if (callback) { try { // console.log("Handling '" + msg_type + "' message: ", msg); callback(fig, msg); } catch (e) { console.log("Exception inside the 'handler_" + msg_type + "' callback:", e, e.stack, msg); } } }; } // from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas mpl.findpos = function(e) { //this section is from http://www.quirksmode.org/js/events_properties.html var targ; if (!e) e = window.event; if (e.target) targ = e.target; else if (e.srcElement) targ = e.srcElement; if (targ.nodeType == 3) // defeat Safari bug targ = targ.parentNode; // jQuery normalizes the pageX and pageY // pageX,Y are the mouse positions relative to the document // offset() returns the position of the element relative to the document var x = e.pageX - $(targ).offset().left; var y = e.pageY - $(targ).offset().top; return {"x": x, "y": y}; }; /* * return a copy of an object with only non-object keys * we need this to avoid circular references * http://stackoverflow.com/a/24161582/3208463 */ function simpleKeys (original) { return Object.keys(original).reduce(function (obj, key) { if (typeof original[key] !== 'object') obj[key] = original[key] return obj; }, {}); } mpl.figure.prototype.mouse_event = function(event, name) { var canvas_pos = mpl.findpos(event) if (name === 'button_press') { this.canvas.focus(); this.canvas_div.focus(); } var x = canvas_pos.x; var y = canvas_pos.y; this.send_message(name, {x: x, y: y, button: event.button, step: event.step, guiEvent: simpleKeys(event)}); /* This prevents the web browser from automatically changing to * the text insertion cursor when the button is pressed. We want * to control all of the cursor setting manually through the * 'cursor' event from matplotlib */ event.preventDefault(); return false; } mpl.figure.prototype._key_event_extra = function(event, name) { // Handle any extra behaviour associated with a key event } mpl.figure.prototype.key_event = function(event, name) { // Prevent repeat events if (name == 'key_press') { if (event.which === this._key) return; else this._key = event.which; } if (name == 'key_release') this._key = null; var value = ''; if (event.ctrlKey && event.which != 17) value += "ctrl+"; if (event.altKey && event.which != 18) value += "alt+"; if (event.shiftKey && event.which != 16) value += "shift+"; value += 'k'; value += event.which.toString(); this._key_event_extra(event, name); this.send_message(name, {key: value, guiEvent: simpleKeys(event)}); return false; } mpl.figure.prototype.toolbar_button_onclick = function(name) { if (name == 'download') { this.handle_save(this, null); } else { this.send_message("toolbar_button", {name: name}); } }; mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) { this.message.textContent = tooltip; }; mpl.toolbar_items = [["Home", "Reset original view", "fa fa-home icon-home", "home"], ["Back", "Back to previous view", "fa fa-arrow-left icon-arrow-left", "back"], ["Forward", "Forward to next view", "fa fa-arrow-right icon-arrow-right", "forward"], ["", "", "", ""], ["Pan", "Pan axes with left mouse, zoom with right", "fa fa-arrows icon-move", "pan"], ["Zoom", "Zoom to rectangle", "fa fa-square-o icon-check-empty", "zoom"], ["", "", "", ""], ["Download", "Download plot", "fa fa-floppy-o icon-save", "download"]]; mpl.extensions = ["eps", "pdf", "png", "ps", "raw", "svg"]; mpl.default_extension = "png";var comm_websocket_adapter = function(comm) { // Create a "websocket"-like object which calls the given IPython comm // object with the appropriate methods. Currently this is a non binary // socket, so there is still some room for performance tuning. var ws = {}; ws.close = function() { comm.close() }; ws.send = function(m) { //console.log('sending', m); comm.send(m); }; // Register the callback with on_msg. comm.on_msg(function(msg) { //console.log('receiving', msg['content']['data'], msg); // Pass the mpl event to the overriden (by mpl) onmessage function. ws.onmessage(msg['content']['data']) }); return ws; } mpl.mpl_figure_comm = function(comm, msg) { // This is the function which gets called when the mpl process // starts-up an IPython Comm through the "matplotlib" channel. var id = msg.content.data.id; // Get hold of the div created by the display call when the Comm // socket was opened in Python. var element = $("#" + id); var ws_proxy = comm_websocket_adapter(comm) function ondownload(figure, format) { window.open(figure.imageObj.src); } var fig = new mpl.figure(id, ws_proxy, ondownload, element.get(0)); // Call onopen now - mpl needs it, as it is assuming we've passed it a real // web socket which is closed, not our websocket->open comm proxy. ws_proxy.onopen(); fig.parent_element = element.get(0); fig.cell_info = mpl.find_output_cell("<div id='" + id + "'></div>"); if (!fig.cell_info) { console.error("Failed to find cell for figure", id, fig); return; } var output_index = fig.cell_info[2] var cell = fig.cell_info[0]; }; mpl.figure.prototype.handle_close = function(fig, msg) { fig.root.unbind('remove') // Update the output cell to use the data from the current canvas. fig.push_to_output(); var dataURL = fig.canvas.toDataURL(); // Re-enable the keyboard manager in IPython - without this line, in FF, // the notebook keyboard shortcuts fail. IPython.keyboard_manager.enable() $(fig.parent_element).html('<img src="' + dataURL + '">'); fig.close_ws(fig, msg); } mpl.figure.prototype.close_ws = function(fig, msg){ fig.send_message('closing', msg); // fig.ws.close() } mpl.figure.prototype.push_to_output = function(remove_interactive) { // Turn the data on the canvas into data in the output cell. var dataURL = this.canvas.toDataURL(); this.cell_info[1]['text/html'] = '<img src="' + dataURL + '">'; } mpl.figure.prototype.updated_canvas_event = function() { // Tell IPython that the notebook contents must change. IPython.notebook.set_dirty(true); this.send_message("ack", {}); var fig = this; // Wait a second, then push the new image to the DOM so // that it is saved nicely (might be nice to debounce this). setTimeout(function () { fig.push_to_output() }, 1000); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items){ var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { continue; }; var button = $('<button class="btn btn-default" href="#" title="' + name + '"><i class="fa ' + image + ' fa-lg"></i></button>'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); nav_element.append(button); } // Add the status bar. var status_bar = $('<span class="mpl-message" style="text-align:right; float: right;"/>'); nav_element.append(status_bar); this.message = status_bar[0]; // Add the close button to the window. var buttongrp = $('<div class="btn-group inline pull-right"></div>'); var button = $('<button class="btn btn-mini btn-primary" href="#" title="Stop Interaction"><i class="fa fa-power-off icon-remove icon-large"></i></button>'); button.click(function (evt) { fig.handle_close(fig, {}); } ); button.mouseover('Stop Interaction', toolbar_mouse_event); buttongrp.append(button); var titlebar = this.root.find($('.ui-dialog-titlebar')); titlebar.prepend(buttongrp); } mpl.figure.prototype._root_extra_style = function(el){ var fig = this el.on("remove", function(){ fig.close_ws(fig, {}); }); } mpl.figure.prototype._canvas_extra_style = function(el){ // this is important to make the div 'focusable el.attr('tabindex', 0) // reach out to IPython and tell the keyboard manager to turn it's self // off when our div gets focus // location in version 3 if (IPython.notebook.keyboard_manager) { IPython.notebook.keyboard_manager.register_events(el); } else { // location in version 2 IPython.keyboard_manager.register_events(el); } } mpl.figure.prototype._key_event_extra = function(event, name) { var manager = IPython.notebook.keyboard_manager; if (!manager) manager = IPython.keyboard_manager; // Check for shift+enter if (event.shiftKey && event.which == 13) { this.canvas_div.blur(); event.shiftKey = false; // Send a "J" for go to next cell event.which = 74; event.keyCode = 74; manager.command_mode(); manager.handle_keydown(event); } } mpl.figure.prototype.handle_save = function(fig, msg) { fig.ondownload(fig, null); } mpl.find_output_cell = function(html_output) { // Return the cell and output element which can be found *uniquely* in the notebook. // Note - this is a bit hacky, but it is done because the "notebook_saving.Notebook" // IPython event is triggered only after the cells have been serialised, which for // our purposes (turning an active figure into a static one), is too late. var cells = IPython.notebook.get_cells(); var ncells = cells.length; for (var i=0; i<ncells; i++) { var cell = cells[i]; if (cell.cell_type === 'code'){ for (var j=0; j<cell.output_area.outputs.length; j++) { var data = cell.output_area.outputs[j]; if (data.data) { // IPython >= 3 moved mimebundle to data attribute of output data = data.data; } if (data['text/html'] == html_output) { return [cell, data, j]; } } } } } // Register the function which deals with the matplotlib target/channel. // The kernel may be null if the page has been refreshed. if (IPython.notebook.kernel != null) { IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm); } </script> </div> </div> <div class="output_area"><div class="prompt"></div> <div class="output_html rendered_html output_subarea "> <img src=""> </div> </div> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[6]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="n">season_ratings</span><span class="p">(</span><span class="s1">&#39;Roswell&#39;</span><span class="p">)</span> </pre></div> </div> </div> </div> <div class="output_wrapper"> <div class="output"> <div class="output_area"><div class="prompt"></div> <div class="output_subarea output_javascript "> <script type="text/javascript"> /* Put everything inside the global mpl namespace */ window.mpl = {}; mpl.get_websocket_type = function() { if (typeof(WebSocket) !== 'undefined') { return WebSocket; } else if (typeof(MozWebSocket) !== 'undefined') { return MozWebSocket; } else { alert('Your browser does not have WebSocket support.' + 'Please try Chrome, Safari or Firefox ≥ 6. ' + 'Firefox 4 and 5 are also supported but you ' + 'have to enable WebSockets in about:config.'); }; } mpl.figure = function(figure_id, websocket, ondownload, parent_element) { this.id = figure_id; this.ws = websocket; this.supports_binary = (this.ws.binaryType != undefined); if (!this.supports_binary) { var warnings = document.getElementById("mpl-warnings"); if (warnings) { warnings.style.display = 'block'; warnings.textContent = ( "This browser does not support binary websocket messages. " + "Performance may be slow."); } } this.imageObj = new Image(); this.context = undefined; this.message = undefined; this.canvas = undefined; this.rubberband_canvas = undefined; this.rubberband_context = undefined; this.format_dropdown = undefined; this.image_mode = 'full'; this.root = $('<div/>'); this._root_extra_style(this.root) this.root.attr('style', 'display: inline-block'); $(parent_element).append(this.root); this._init_header(this); this._init_canvas(this); this._init_toolbar(this); var fig = this; this.waiting = false; this.ws.onopen = function () { fig.send_message("supports_binary", {value: fig.supports_binary}); fig.send_message("send_image_mode", {}); fig.send_message("refresh", {}); } this.imageObj.onload = function() { if (fig.image_mode == 'full') { // Full images could contain transparency (where diff images // almost always do), so we need to clear the canvas so that // there is no ghosting. fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height); } fig.context.drawImage(fig.imageObj, 0, 0); }; this.imageObj.onunload = function() { this.ws.close(); } this.ws.onmessage = this._make_on_message_function(this); this.ondownload = ondownload; } mpl.figure.prototype._init_header = function() { var titlebar = $( '<div class="ui-dialog-titlebar ui-widget-header ui-corner-all ' + 'ui-helper-clearfix"/>'); var titletext = $( '<div class="ui-dialog-title" style="width: 100%; ' + 'text-align: center; padding: 3px;"/>'); titlebar.append(titletext) this.root.append(titlebar); this.header = titletext[0]; } mpl.figure.prototype._canvas_extra_style = function(canvas_div) { } mpl.figure.prototype._root_extra_style = function(canvas_div) { } mpl.figure.prototype._init_canvas = function() { var fig = this; var canvas_div = $('<div/>'); canvas_div.attr('style', 'position: relative; clear: both; outline: 0'); function canvas_keyboard_event(event) { return fig.key_event(event, event['data']); } canvas_div.keydown('key_press', canvas_keyboard_event); canvas_div.keyup('key_release', canvas_keyboard_event); this.canvas_div = canvas_div this._canvas_extra_style(canvas_div) this.root.append(canvas_div); var canvas = $('<canvas/>'); canvas.addClass('mpl-canvas'); canvas.attr('style', "left: 0; top: 0; z-index: 0; outline: 0") this.canvas = canvas[0]; this.context = canvas[0].getContext("2d"); var rubberband = $('<canvas/>'); rubberband.attr('style', "position: absolute; left: 0; top: 0; z-index: 1;") var pass_mouse_events = true; canvas_div.resizable({ start: function(event, ui) { pass_mouse_events = false; }, resize: function(event, ui) { fig.request_resize(ui.size.width, ui.size.height); }, stop: function(event, ui) { pass_mouse_events = true; fig.request_resize(ui.size.width, ui.size.height); }, }); function mouse_event_fn(event) { if (pass_mouse_events) return fig.mouse_event(event, event['data']); } rubberband.mousedown('button_press', mouse_event_fn); rubberband.mouseup('button_release', mouse_event_fn); // Throttle sequential mouse events to 1 every 20ms. rubberband.mousemove('motion_notify', mouse_event_fn); rubberband.mouseenter('figure_enter', mouse_event_fn); rubberband.mouseleave('figure_leave', mouse_event_fn); canvas_div.on("wheel", function (event) { event = event.originalEvent; event['data'] = 'scroll' if (event.deltaY < 0) { event.step = 1; } else { event.step = -1; } mouse_event_fn(event); }); canvas_div.append(canvas); canvas_div.append(rubberband); this.rubberband = rubberband; this.rubberband_canvas = rubberband[0]; this.rubberband_context = rubberband[0].getContext("2d"); this.rubberband_context.strokeStyle = "#000000"; this._resize_canvas = function(width, height) { // Keep the size of the canvas, canvas container, and rubber band // canvas in synch. canvas_div.css('width', width) canvas_div.css('height', height) canvas.attr('width', width); canvas.attr('height', height); rubberband.attr('width', width); rubberband.attr('height', height); } // Set the figure to an initial 600x600px, this will subsequently be updated // upon first draw. this._resize_canvas(600, 600); // Disable right mouse context menu. $(this.rubberband_canvas).bind("contextmenu",function(e){ return false; }); function set_focus () { canvas.focus(); canvas_div.focus(); } window.setTimeout(set_focus, 100); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items) { var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { // put a spacer in here. continue; } var button = $('<button/>'); button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' + 'ui-button-icon-only'); button.attr('role', 'button'); button.attr('aria-disabled', 'false'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); var icon_img = $('<span/>'); icon_img.addClass('ui-button-icon-primary ui-icon'); icon_img.addClass(image); icon_img.addClass('ui-corner-all'); var tooltip_span = $('<span/>'); tooltip_span.addClass('ui-button-text'); tooltip_span.html(tooltip); button.append(icon_img); button.append(tooltip_span); nav_element.append(button); } var fmt_picker_span = $('<span/>'); var fmt_picker = $('<select/>'); fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content'); fmt_picker_span.append(fmt_picker); nav_element.append(fmt_picker_span); this.format_dropdown = fmt_picker[0]; for (var ind in mpl.extensions) { var fmt = mpl.extensions[ind]; var option = $( '<option/>', {selected: fmt === mpl.default_extension}).html(fmt); fmt_picker.append(option) } // Add hover states to the ui-buttons $( ".ui-button" ).hover( function() { $(this).addClass("ui-state-hover");}, function() { $(this).removeClass("ui-state-hover");} ); var status_bar = $('<span class="mpl-message"/>'); nav_element.append(status_bar); this.message = status_bar[0]; } mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) { // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client, // which will in turn request a refresh of the image. this.send_message('resize', {'width': x_pixels, 'height': y_pixels}); } mpl.figure.prototype.send_message = function(type, properties) { properties['type'] = type; properties['figure_id'] = this.id; this.ws.send(JSON.stringify(properties)); } mpl.figure.prototype.send_draw_message = function() { if (!this.waiting) { this.waiting = true; this.ws.send(JSON.stringify({type: "draw", figure_id: this.id})); } } mpl.figure.prototype.handle_save = function(fig, msg) { var format_dropdown = fig.format_dropdown; var format = format_dropdown.options[format_dropdown.selectedIndex].value; fig.ondownload(fig, format); } mpl.figure.prototype.handle_resize = function(fig, msg) { var size = msg['size']; if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) { fig._resize_canvas(size[0], size[1]); fig.send_message("refresh", {}); }; } mpl.figure.prototype.handle_rubberband = function(fig, msg) { var x0 = msg['x0']; var y0 = fig.canvas.height - msg['y0']; var x1 = msg['x1']; var y1 = fig.canvas.height - msg['y1']; x0 = Math.floor(x0) + 0.5; y0 = Math.floor(y0) + 0.5; x1 = Math.floor(x1) + 0.5; y1 = Math.floor(y1) + 0.5; var min_x = Math.min(x0, x1); var min_y = Math.min(y0, y1); var width = Math.abs(x1 - x0); var height = Math.abs(y1 - y0); fig.rubberband_context.clearRect( 0, 0, fig.canvas.width, fig.canvas.height); fig.rubberband_context.strokeRect(min_x, min_y, width, height); } mpl.figure.prototype.handle_figure_label = function(fig, msg) { // Updates the figure title. fig.header.textContent = msg['label']; } mpl.figure.prototype.handle_cursor = function(fig, msg) { var cursor = msg['cursor']; switch(cursor) { case 0: cursor = 'pointer'; break; case 1: cursor = 'default'; break; case 2: cursor = 'crosshair'; break; case 3: cursor = 'move'; break; } fig.rubberband_canvas.style.cursor = cursor; } mpl.figure.prototype.handle_message = function(fig, msg) { fig.message.textContent = msg['message']; } mpl.figure.prototype.handle_draw = function(fig, msg) { // Request the server to send over a new figure. fig.send_draw_message(); } mpl.figure.prototype.handle_image_mode = function(fig, msg) { fig.image_mode = msg['mode']; } mpl.figure.prototype.updated_canvas_event = function() { // Called whenever the canvas gets updated. this.send_message("ack", {}); } // A function to construct a web socket function for onmessage handling. // Called in the figure constructor. mpl.figure.prototype._make_on_message_function = function(fig) { return function socket_on_message(evt) { if (evt.data instanceof Blob) { /* FIXME: We get "Resource interpreted as Image but * transferred with MIME type text/plain:" errors on * Chrome. But how to set the MIME type? It doesn't seem * to be part of the websocket stream */ evt.data.type = "image/png"; /* Free the memory for the previous frames */ if (fig.imageObj.src) { (window.URL || window.webkitURL).revokeObjectURL( fig.imageObj.src); } fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL( evt.data); fig.updated_canvas_event(); fig.waiting = false; return; } else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == "data:image/png;base64") { fig.imageObj.src = evt.data; fig.updated_canvas_event(); fig.waiting = false; return; } var msg = JSON.parse(evt.data); var msg_type = msg['type']; // Call the "handle_{type}" callback, which takes // the figure and JSON message as its only arguments. try { var callback = fig["handle_" + msg_type]; } catch (e) { console.log("No handler for the '" + msg_type + "' message type: ", msg); return; } if (callback) { try { // console.log("Handling '" + msg_type + "' message: ", msg); callback(fig, msg); } catch (e) { console.log("Exception inside the 'handler_" + msg_type + "' callback:", e, e.stack, msg); } } }; } // from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas mpl.findpos = function(e) { //this section is from http://www.quirksmode.org/js/events_properties.html var targ; if (!e) e = window.event; if (e.target) targ = e.target; else if (e.srcElement) targ = e.srcElement; if (targ.nodeType == 3) // defeat Safari bug targ = targ.parentNode; // jQuery normalizes the pageX and pageY // pageX,Y are the mouse positions relative to the document // offset() returns the position of the element relative to the document var x = e.pageX - $(targ).offset().left; var y = e.pageY - $(targ).offset().top; return {"x": x, "y": y}; }; /* * return a copy of an object with only non-object keys * we need this to avoid circular references * http://stackoverflow.com/a/24161582/3208463 */ function simpleKeys (original) { return Object.keys(original).reduce(function (obj, key) { if (typeof original[key] !== 'object') obj[key] = original[key] return obj; }, {}); } mpl.figure.prototype.mouse_event = function(event, name) { var canvas_pos = mpl.findpos(event) if (name === 'button_press') { this.canvas.focus(); this.canvas_div.focus(); } var x = canvas_pos.x; var y = canvas_pos.y; this.send_message(name, {x: x, y: y, button: event.button, step: event.step, guiEvent: simpleKeys(event)}); /* This prevents the web browser from automatically changing to * the text insertion cursor when the button is pressed. We want * to control all of the cursor setting manually through the * 'cursor' event from matplotlib */ event.preventDefault(); return false; } mpl.figure.prototype._key_event_extra = function(event, name) { // Handle any extra behaviour associated with a key event } mpl.figure.prototype.key_event = function(event, name) { // Prevent repeat events if (name == 'key_press') { if (event.which === this._key) return; else this._key = event.which; } if (name == 'key_release') this._key = null; var value = ''; if (event.ctrlKey && event.which != 17) value += "ctrl+"; if (event.altKey && event.which != 18) value += "alt+"; if (event.shiftKey && event.which != 16) value += "shift+"; value += 'k'; value += event.which.toString(); this._key_event_extra(event, name); this.send_message(name, {key: value, guiEvent: simpleKeys(event)}); return false; } mpl.figure.prototype.toolbar_button_onclick = function(name) { if (name == 'download') { this.handle_save(this, null); } else { this.send_message("toolbar_button", {name: name}); } }; mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) { this.message.textContent = tooltip; }; mpl.toolbar_items = [["Home", "Reset original view", "fa fa-home icon-home", "home"], ["Back", "Back to previous view", "fa fa-arrow-left icon-arrow-left", "back"], ["Forward", "Forward to next view", "fa fa-arrow-right icon-arrow-right", "forward"], ["", "", "", ""], ["Pan", "Pan axes with left mouse, zoom with right", "fa fa-arrows icon-move", "pan"], ["Zoom", "Zoom to rectangle", "fa fa-square-o icon-check-empty", "zoom"], ["", "", "", ""], ["Download", "Download plot", "fa fa-floppy-o icon-save", "download"]]; mpl.extensions = ["eps", "pdf", "png", "ps", "raw", "svg"]; mpl.default_extension = "png";var comm_websocket_adapter = function(comm) { // Create a "websocket"-like object which calls the given IPython comm // object with the appropriate methods. Currently this is a non binary // socket, so there is still some room for performance tuning. var ws = {}; ws.close = function() { comm.close() }; ws.send = function(m) { //console.log('sending', m); comm.send(m); }; // Register the callback with on_msg. comm.on_msg(function(msg) { //console.log('receiving', msg['content']['data'], msg); // Pass the mpl event to the overriden (by mpl) onmessage function. ws.onmessage(msg['content']['data']) }); return ws; } mpl.mpl_figure_comm = function(comm, msg) { // This is the function which gets called when the mpl process // starts-up an IPython Comm through the "matplotlib" channel. var id = msg.content.data.id; // Get hold of the div created by the display call when the Comm // socket was opened in Python. var element = $("#" + id); var ws_proxy = comm_websocket_adapter(comm) function ondownload(figure, format) { window.open(figure.imageObj.src); } var fig = new mpl.figure(id, ws_proxy, ondownload, element.get(0)); // Call onopen now - mpl needs it, as it is assuming we've passed it a real // web socket which is closed, not our websocket->open comm proxy. ws_proxy.onopen(); fig.parent_element = element.get(0); fig.cell_info = mpl.find_output_cell("<div id='" + id + "'></div>"); if (!fig.cell_info) { console.error("Failed to find cell for figure", id, fig); return; } var output_index = fig.cell_info[2] var cell = fig.cell_info[0]; }; mpl.figure.prototype.handle_close = function(fig, msg) { fig.root.unbind('remove') // Update the output cell to use the data from the current canvas. fig.push_to_output(); var dataURL = fig.canvas.toDataURL(); // Re-enable the keyboard manager in IPython - without this line, in FF, // the notebook keyboard shortcuts fail. IPython.keyboard_manager.enable() $(fig.parent_element).html('<img src="' + dataURL + '">'); fig.close_ws(fig, msg); } mpl.figure.prototype.close_ws = function(fig, msg){ fig.send_message('closing', msg); // fig.ws.close() } mpl.figure.prototype.push_to_output = function(remove_interactive) { // Turn the data on the canvas into data in the output cell. var dataURL = this.canvas.toDataURL(); this.cell_info[1]['text/html'] = '<img src="' + dataURL + '">'; } mpl.figure.prototype.updated_canvas_event = function() { // Tell IPython that the notebook contents must change. IPython.notebook.set_dirty(true); this.send_message("ack", {}); var fig = this; // Wait a second, then push the new image to the DOM so // that it is saved nicely (might be nice to debounce this). setTimeout(function () { fig.push_to_output() }, 1000); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items){ var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { continue; }; var button = $('<button class="btn btn-default" href="#" title="' + name + '"><i class="fa ' + image + ' fa-lg"></i></button>'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); nav_element.append(button); } // Add the status bar. var status_bar = $('<span class="mpl-message" style="text-align:right; float: right;"/>'); nav_element.append(status_bar); this.message = status_bar[0]; // Add the close button to the window. var buttongrp = $('<div class="btn-group inline pull-right"></div>'); var button = $('<button class="btn btn-mini btn-primary" href="#" title="Stop Interaction"><i class="fa fa-power-off icon-remove icon-large"></i></button>'); button.click(function (evt) { fig.handle_close(fig, {}); } ); button.mouseover('Stop Interaction', toolbar_mouse_event); buttongrp.append(button); var titlebar = this.root.find($('.ui-dialog-titlebar')); titlebar.prepend(buttongrp); } mpl.figure.prototype._root_extra_style = function(el){ var fig = this el.on("remove", function(){ fig.close_ws(fig, {}); }); } mpl.figure.prototype._canvas_extra_style = function(el){ // this is important to make the div 'focusable el.attr('tabindex', 0) // reach out to IPython and tell the keyboard manager to turn it's self // off when our div gets focus // location in version 3 if (IPython.notebook.keyboard_manager) { IPython.notebook.keyboard_manager.register_events(el); } else { // location in version 2 IPython.keyboard_manager.register_events(el); } } mpl.figure.prototype._key_event_extra = function(event, name) { var manager = IPython.notebook.keyboard_manager; if (!manager) manager = IPython.keyboard_manager; // Check for shift+enter if (event.shiftKey && event.which == 13) { this.canvas_div.blur(); event.shiftKey = false; // Send a "J" for go to next cell event.which = 74; event.keyCode = 74; manager.command_mode(); manager.handle_keydown(event); } } mpl.figure.prototype.handle_save = function(fig, msg) { fig.ondownload(fig, null); } mpl.find_output_cell = function(html_output) { // Return the cell and output element which can be found *uniquely* in the notebook. // Note - this is a bit hacky, but it is done because the "notebook_saving.Notebook" // IPython event is triggered only after the cells have been serialised, which for // our purposes (turning an active figure into a static one), is too late. var cells = IPython.notebook.get_cells(); var ncells = cells.length; for (var i=0; i<ncells; i++) { var cell = cells[i]; if (cell.cell_type === 'code'){ for (var j=0; j<cell.output_area.outputs.length; j++) { var data = cell.output_area.outputs[j]; if (data.data) { // IPython >= 3 moved mimebundle to data attribute of output data = data.data; } if (data['text/html'] == html_output) { return [cell, data, j]; } } } } } // Register the function which deals with the matplotlib target/channel. // The kernel may be null if the page has been refreshed. if (IPython.notebook.kernel != null) { IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm); } </script> </div> </div> <div class="output_area"><div class="prompt"></div> <div class="output_html rendered_html output_subarea "> <img src=""> </div> </div> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[7]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="n">season_ratings</span><span class="p">(</span><span class="s1">&#39;Veronica Mars&#39;</span><span class="p">)</span> </pre></div> </div> </div> </div> <div class="output_wrapper"> <div class="output"> <div class="output_area"><div class="prompt"></div> <div class="output_subarea output_javascript "> <script type="text/javascript"> /* Put everything inside the global mpl namespace */ window.mpl = {}; mpl.get_websocket_type = function() { if (typeof(WebSocket) !== 'undefined') { return WebSocket; } else if (typeof(MozWebSocket) !== 'undefined') { return MozWebSocket; } else { alert('Your browser does not have WebSocket support.' + 'Please try Chrome, Safari or Firefox ≥ 6. ' + 'Firefox 4 and 5 are also supported but you ' + 'have to enable WebSockets in about:config.'); }; } mpl.figure = function(figure_id, websocket, ondownload, parent_element) { this.id = figure_id; this.ws = websocket; this.supports_binary = (this.ws.binaryType != undefined); if (!this.supports_binary) { var warnings = document.getElementById("mpl-warnings"); if (warnings) { warnings.style.display = 'block'; warnings.textContent = ( "This browser does not support binary websocket messages. " + "Performance may be slow."); } } this.imageObj = new Image(); this.context = undefined; this.message = undefined; this.canvas = undefined; this.rubberband_canvas = undefined; this.rubberband_context = undefined; this.format_dropdown = undefined; this.image_mode = 'full'; this.root = $('<div/>'); this._root_extra_style(this.root) this.root.attr('style', 'display: inline-block'); $(parent_element).append(this.root); this._init_header(this); this._init_canvas(this); this._init_toolbar(this); var fig = this; this.waiting = false; this.ws.onopen = function () { fig.send_message("supports_binary", {value: fig.supports_binary}); fig.send_message("send_image_mode", {}); fig.send_message("refresh", {}); } this.imageObj.onload = function() { if (fig.image_mode == 'full') { // Full images could contain transparency (where diff images // almost always do), so we need to clear the canvas so that // there is no ghosting. fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height); } fig.context.drawImage(fig.imageObj, 0, 0); }; this.imageObj.onunload = function() { this.ws.close(); } this.ws.onmessage = this._make_on_message_function(this); this.ondownload = ondownload; } mpl.figure.prototype._init_header = function() { var titlebar = $( '<div class="ui-dialog-titlebar ui-widget-header ui-corner-all ' + 'ui-helper-clearfix"/>'); var titletext = $( '<div class="ui-dialog-title" style="width: 100%; ' + 'text-align: center; padding: 3px;"/>'); titlebar.append(titletext) this.root.append(titlebar); this.header = titletext[0]; } mpl.figure.prototype._canvas_extra_style = function(canvas_div) { } mpl.figure.prototype._root_extra_style = function(canvas_div) { } mpl.figure.prototype._init_canvas = function() { var fig = this; var canvas_div = $('<div/>'); canvas_div.attr('style', 'position: relative; clear: both; outline: 0'); function canvas_keyboard_event(event) { return fig.key_event(event, event['data']); } canvas_div.keydown('key_press', canvas_keyboard_event); canvas_div.keyup('key_release', canvas_keyboard_event); this.canvas_div = canvas_div this._canvas_extra_style(canvas_div) this.root.append(canvas_div); var canvas = $('<canvas/>'); canvas.addClass('mpl-canvas'); canvas.attr('style', "left: 0; top: 0; z-index: 0; outline: 0") this.canvas = canvas[0]; this.context = canvas[0].getContext("2d"); var rubberband = $('<canvas/>'); rubberband.attr('style', "position: absolute; left: 0; top: 0; z-index: 1;") var pass_mouse_events = true; canvas_div.resizable({ start: function(event, ui) { pass_mouse_events = false; }, resize: function(event, ui) { fig.request_resize(ui.size.width, ui.size.height); }, stop: function(event, ui) { pass_mouse_events = true; fig.request_resize(ui.size.width, ui.size.height); }, }); function mouse_event_fn(event) { if (pass_mouse_events) return fig.mouse_event(event, event['data']); } rubberband.mousedown('button_press', mouse_event_fn); rubberband.mouseup('button_release', mouse_event_fn); // Throttle sequential mouse events to 1 every 20ms. rubberband.mousemove('motion_notify', mouse_event_fn); rubberband.mouseenter('figure_enter', mouse_event_fn); rubberband.mouseleave('figure_leave', mouse_event_fn); canvas_div.on("wheel", function (event) { event = event.originalEvent; event['data'] = 'scroll' if (event.deltaY < 0) { event.step = 1; } else { event.step = -1; } mouse_event_fn(event); }); canvas_div.append(canvas); canvas_div.append(rubberband); this.rubberband = rubberband; this.rubberband_canvas = rubberband[0]; this.rubberband_context = rubberband[0].getContext("2d"); this.rubberband_context.strokeStyle = "#000000"; this._resize_canvas = function(width, height) { // Keep the size of the canvas, canvas container, and rubber band // canvas in synch. canvas_div.css('width', width) canvas_div.css('height', height) canvas.attr('width', width); canvas.attr('height', height); rubberband.attr('width', width); rubberband.attr('height', height); } // Set the figure to an initial 600x600px, this will subsequently be updated // upon first draw. this._resize_canvas(600, 600); // Disable right mouse context menu. $(this.rubberband_canvas).bind("contextmenu",function(e){ return false; }); function set_focus () { canvas.focus(); canvas_div.focus(); } window.setTimeout(set_focus, 100); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items) { var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { // put a spacer in here. continue; } var button = $('<button/>'); button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' + 'ui-button-icon-only'); button.attr('role', 'button'); button.attr('aria-disabled', 'false'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); var icon_img = $('<span/>'); icon_img.addClass('ui-button-icon-primary ui-icon'); icon_img.addClass(image); icon_img.addClass('ui-corner-all'); var tooltip_span = $('<span/>'); tooltip_span.addClass('ui-button-text'); tooltip_span.html(tooltip); button.append(icon_img); button.append(tooltip_span); nav_element.append(button); } var fmt_picker_span = $('<span/>'); var fmt_picker = $('<select/>'); fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content'); fmt_picker_span.append(fmt_picker); nav_element.append(fmt_picker_span); this.format_dropdown = fmt_picker[0]; for (var ind in mpl.extensions) { var fmt = mpl.extensions[ind]; var option = $( '<option/>', {selected: fmt === mpl.default_extension}).html(fmt); fmt_picker.append(option) } // Add hover states to the ui-buttons $( ".ui-button" ).hover( function() { $(this).addClass("ui-state-hover");}, function() { $(this).removeClass("ui-state-hover");} ); var status_bar = $('<span class="mpl-message"/>'); nav_element.append(status_bar); this.message = status_bar[0]; } mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) { // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client, // which will in turn request a refresh of the image. this.send_message('resize', {'width': x_pixels, 'height': y_pixels}); } mpl.figure.prototype.send_message = function(type, properties) { properties['type'] = type; properties['figure_id'] = this.id; this.ws.send(JSON.stringify(properties)); } mpl.figure.prototype.send_draw_message = function() { if (!this.waiting) { this.waiting = true; this.ws.send(JSON.stringify({type: "draw", figure_id: this.id})); } } mpl.figure.prototype.handle_save = function(fig, msg) { var format_dropdown = fig.format_dropdown; var format = format_dropdown.options[format_dropdown.selectedIndex].value; fig.ondownload(fig, format); } mpl.figure.prototype.handle_resize = function(fig, msg) { var size = msg['size']; if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) { fig._resize_canvas(size[0], size[1]); fig.send_message("refresh", {}); }; } mpl.figure.prototype.handle_rubberband = function(fig, msg) { var x0 = msg['x0']; var y0 = fig.canvas.height - msg['y0']; var x1 = msg['x1']; var y1 = fig.canvas.height - msg['y1']; x0 = Math.floor(x0) + 0.5; y0 = Math.floor(y0) + 0.5; x1 = Math.floor(x1) + 0.5; y1 = Math.floor(y1) + 0.5; var min_x = Math.min(x0, x1); var min_y = Math.min(y0, y1); var width = Math.abs(x1 - x0); var height = Math.abs(y1 - y0); fig.rubberband_context.clearRect( 0, 0, fig.canvas.width, fig.canvas.height); fig.rubberband_context.strokeRect(min_x, min_y, width, height); } mpl.figure.prototype.handle_figure_label = function(fig, msg) { // Updates the figure title. fig.header.textContent = msg['label']; } mpl.figure.prototype.handle_cursor = function(fig, msg) { var cursor = msg['cursor']; switch(cursor) { case 0: cursor = 'pointer'; break; case 1: cursor = 'default'; break; case 2: cursor = 'crosshair'; break; case 3: cursor = 'move'; break; } fig.rubberband_canvas.style.cursor = cursor; } mpl.figure.prototype.handle_message = function(fig, msg) { fig.message.textContent = msg['message']; } mpl.figure.prototype.handle_draw = function(fig, msg) { // Request the server to send over a new figure. fig.send_draw_message(); } mpl.figure.prototype.handle_image_mode = function(fig, msg) { fig.image_mode = msg['mode']; } mpl.figure.prototype.updated_canvas_event = function() { // Called whenever the canvas gets updated. this.send_message("ack", {}); } // A function to construct a web socket function for onmessage handling. // Called in the figure constructor. mpl.figure.prototype._make_on_message_function = function(fig) { return function socket_on_message(evt) { if (evt.data instanceof Blob) { /* FIXME: We get "Resource interpreted as Image but * transferred with MIME type text/plain:" errors on * Chrome. But how to set the MIME type? It doesn't seem * to be part of the websocket stream */ evt.data.type = "image/png"; /* Free the memory for the previous frames */ if (fig.imageObj.src) { (window.URL || window.webkitURL).revokeObjectURL( fig.imageObj.src); } fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL( evt.data); fig.updated_canvas_event(); fig.waiting = false; return; } else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == "data:image/png;base64") { fig.imageObj.src = evt.data; fig.updated_canvas_event(); fig.waiting = false; return; } var msg = JSON.parse(evt.data); var msg_type = msg['type']; // Call the "handle_{type}" callback, which takes // the figure and JSON message as its only arguments. try { var callback = fig["handle_" + msg_type]; } catch (e) { console.log("No handler for the '" + msg_type + "' message type: ", msg); return; } if (callback) { try { // console.log("Handling '" + msg_type + "' message: ", msg); callback(fig, msg); } catch (e) { console.log("Exception inside the 'handler_" + msg_type + "' callback:", e, e.stack, msg); } } }; } // from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas mpl.findpos = function(e) { //this section is from http://www.quirksmode.org/js/events_properties.html var targ; if (!e) e = window.event; if (e.target) targ = e.target; else if (e.srcElement) targ = e.srcElement; if (targ.nodeType == 3) // defeat Safari bug targ = targ.parentNode; // jQuery normalizes the pageX and pageY // pageX,Y are the mouse positions relative to the document // offset() returns the position of the element relative to the document var x = e.pageX - $(targ).offset().left; var y = e.pageY - $(targ).offset().top; return {"x": x, "y": y}; }; /* * return a copy of an object with only non-object keys * we need this to avoid circular references * http://stackoverflow.com/a/24161582/3208463 */ function simpleKeys (original) { return Object.keys(original).reduce(function (obj, key) { if (typeof original[key] !== 'object') obj[key] = original[key] return obj; }, {}); } mpl.figure.prototype.mouse_event = function(event, name) { var canvas_pos = mpl.findpos(event) if (name === 'button_press') { this.canvas.focus(); this.canvas_div.focus(); } var x = canvas_pos.x; var y = canvas_pos.y; this.send_message(name, {x: x, y: y, button: event.button, step: event.step, guiEvent: simpleKeys(event)}); /* This prevents the web browser from automatically changing to * the text insertion cursor when the button is pressed. We want * to control all of the cursor setting manually through the * 'cursor' event from matplotlib */ event.preventDefault(); return false; } mpl.figure.prototype._key_event_extra = function(event, name) { // Handle any extra behaviour associated with a key event } mpl.figure.prototype.key_event = function(event, name) { // Prevent repeat events if (name == 'key_press') { if (event.which === this._key) return; else this._key = event.which; } if (name == 'key_release') this._key = null; var value = ''; if (event.ctrlKey && event.which != 17) value += "ctrl+"; if (event.altKey && event.which != 18) value += "alt+"; if (event.shiftKey && event.which != 16) value += "shift+"; value += 'k'; value += event.which.toString(); this._key_event_extra(event, name); this.send_message(name, {key: value, guiEvent: simpleKeys(event)}); return false; } mpl.figure.prototype.toolbar_button_onclick = function(name) { if (name == 'download') { this.handle_save(this, null); } else { this.send_message("toolbar_button", {name: name}); } }; mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) { this.message.textContent = tooltip; }; mpl.toolbar_items = [["Home", "Reset original view", "fa fa-home icon-home", "home"], ["Back", "Back to previous view", "fa fa-arrow-left icon-arrow-left", "back"], ["Forward", "Forward to next view", "fa fa-arrow-right icon-arrow-right", "forward"], ["", "", "", ""], ["Pan", "Pan axes with left mouse, zoom with right", "fa fa-arrows icon-move", "pan"], ["Zoom", "Zoom to rectangle", "fa fa-square-o icon-check-empty", "zoom"], ["", "", "", ""], ["Download", "Download plot", "fa fa-floppy-o icon-save", "download"]]; mpl.extensions = ["eps", "pdf", "png", "ps", "raw", "svg"]; mpl.default_extension = "png";var comm_websocket_adapter = function(comm) { // Create a "websocket"-like object which calls the given IPython comm // object with the appropriate methods. Currently this is a non binary // socket, so there is still some room for performance tuning. var ws = {}; ws.close = function() { comm.close() }; ws.send = function(m) { //console.log('sending', m); comm.send(m); }; // Register the callback with on_msg. comm.on_msg(function(msg) { //console.log('receiving', msg['content']['data'], msg); // Pass the mpl event to the overriden (by mpl) onmessage function. ws.onmessage(msg['content']['data']) }); return ws; } mpl.mpl_figure_comm = function(comm, msg) { // This is the function which gets called when the mpl process // starts-up an IPython Comm through the "matplotlib" channel. var id = msg.content.data.id; // Get hold of the div created by the display call when the Comm // socket was opened in Python. var element = $("#" + id); var ws_proxy = comm_websocket_adapter(comm) function ondownload(figure, format) { window.open(figure.imageObj.src); } var fig = new mpl.figure(id, ws_proxy, ondownload, element.get(0)); // Call onopen now - mpl needs it, as it is assuming we've passed it a real // web socket which is closed, not our websocket->open comm proxy. ws_proxy.onopen(); fig.parent_element = element.get(0); fig.cell_info = mpl.find_output_cell("<div id='" + id + "'></div>"); if (!fig.cell_info) { console.error("Failed to find cell for figure", id, fig); return; } var output_index = fig.cell_info[2] var cell = fig.cell_info[0]; }; mpl.figure.prototype.handle_close = function(fig, msg) { fig.root.unbind('remove') // Update the output cell to use the data from the current canvas. fig.push_to_output(); var dataURL = fig.canvas.toDataURL(); // Re-enable the keyboard manager in IPython - without this line, in FF, // the notebook keyboard shortcuts fail. IPython.keyboard_manager.enable() $(fig.parent_element).html('<img src="' + dataURL + '">'); fig.close_ws(fig, msg); } mpl.figure.prototype.close_ws = function(fig, msg){ fig.send_message('closing', msg); // fig.ws.close() } mpl.figure.prototype.push_to_output = function(remove_interactive) { // Turn the data on the canvas into data in the output cell. var dataURL = this.canvas.toDataURL(); this.cell_info[1]['text/html'] = '<img src="' + dataURL + '">'; } mpl.figure.prototype.updated_canvas_event = function() { // Tell IPython that the notebook contents must change. IPython.notebook.set_dirty(true); this.send_message("ack", {}); var fig = this; // Wait a second, then push the new image to the DOM so // that it is saved nicely (might be nice to debounce this). setTimeout(function () { fig.push_to_output() }, 1000); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items){ var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { continue; }; var button = $('<button class="btn btn-default" href="#" title="' + name + '"><i class="fa ' + image + ' fa-lg"></i></button>'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); nav_element.append(button); } // Add the status bar. var status_bar = $('<span class="mpl-message" style="text-align:right; float: right;"/>'); nav_element.append(status_bar); this.message = status_bar[0]; // Add the close button to the window. var buttongrp = $('<div class="btn-group inline pull-right"></div>'); var button = $('<button class="btn btn-mini btn-primary" href="#" title="Stop Interaction"><i class="fa fa-power-off icon-remove icon-large"></i></button>'); button.click(function (evt) { fig.handle_close(fig, {}); } ); button.mouseover('Stop Interaction', toolbar_mouse_event); buttongrp.append(button); var titlebar = this.root.find($('.ui-dialog-titlebar')); titlebar.prepend(buttongrp); } mpl.figure.prototype._root_extra_style = function(el){ var fig = this el.on("remove", function(){ fig.close_ws(fig, {}); }); } mpl.figure.prototype._canvas_extra_style = function(el){ // this is important to make the div 'focusable el.attr('tabindex', 0) // reach out to IPython and tell the keyboard manager to turn it's self // off when our div gets focus // location in version 3 if (IPython.notebook.keyboard_manager) { IPython.notebook.keyboard_manager.register_events(el); } else { // location in version 2 IPython.keyboard_manager.register_events(el); } } mpl.figure.prototype._key_event_extra = function(event, name) { var manager = IPython.notebook.keyboard_manager; if (!manager) manager = IPython.keyboard_manager; // Check for shift+enter if (event.shiftKey && event.which == 13) { this.canvas_div.blur(); event.shiftKey = false; // Send a "J" for go to next cell event.which = 74; event.keyCode = 74; manager.command_mode(); manager.handle_keydown(event); } } mpl.figure.prototype.handle_save = function(fig, msg) { fig.ondownload(fig, null); } mpl.find_output_cell = function(html_output) { // Return the cell and output element which can be found *uniquely* in the notebook. // Note - this is a bit hacky, but it is done because the "notebook_saving.Notebook" // IPython event is triggered only after the cells have been serialised, which for // our purposes (turning an active figure into a static one), is too late. var cells = IPython.notebook.get_cells(); var ncells = cells.length; for (var i=0; i<ncells; i++) { var cell = cells[i]; if (cell.cell_type === 'code'){ for (var j=0; j<cell.output_area.outputs.length; j++) { var data = cell.output_area.outputs[j]; if (data.data) { // IPython >= 3 moved mimebundle to data attribute of output data = data.data; } if (data['text/html'] == html_output) { return [cell, data, j]; } } } } } // Register the function which deals with the matplotlib target/channel. // The kernel may be null if the page has been refreshed. if (IPython.notebook.kernel != null) { IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm); } </script> </div> </div> <div class="output_area"><div class="prompt"></div> <div class="output_html rendered_html output_subarea "> <img src=""> </div> </div> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[8]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="n">season_ratings</span><span class="p">(</span><span class="s1">&#39;Breaking Bad&#39;</span><span class="p">)</span> </pre></div> </div> </div> </div> <div class="output_wrapper"> <div class="output"> <div class="output_area"><div class="prompt"></div> <div class="output_subarea output_javascript "> <script type="text/javascript"> /* Put everything inside the global mpl namespace */ window.mpl = {}; mpl.get_websocket_type = function() { if (typeof(WebSocket) !== 'undefined') { return WebSocket; } else if (typeof(MozWebSocket) !== 'undefined') { return MozWebSocket; } else { alert('Your browser does not have WebSocket support.' + 'Please try Chrome, Safari or Firefox ≥ 6. ' + 'Firefox 4 and 5 are also supported but you ' + 'have to enable WebSockets in about:config.'); }; } mpl.figure = function(figure_id, websocket, ondownload, parent_element) { this.id = figure_id; this.ws = websocket; this.supports_binary = (this.ws.binaryType != undefined); if (!this.supports_binary) { var warnings = document.getElementById("mpl-warnings"); if (warnings) { warnings.style.display = 'block'; warnings.textContent = ( "This browser does not support binary websocket messages. " + "Performance may be slow."); } } this.imageObj = new Image(); this.context = undefined; this.message = undefined; this.canvas = undefined; this.rubberband_canvas = undefined; this.rubberband_context = undefined; this.format_dropdown = undefined; this.image_mode = 'full'; this.root = $('<div/>'); this._root_extra_style(this.root) this.root.attr('style', 'display: inline-block'); $(parent_element).append(this.root); this._init_header(this); this._init_canvas(this); this._init_toolbar(this); var fig = this; this.waiting = false; this.ws.onopen = function () { fig.send_message("supports_binary", {value: fig.supports_binary}); fig.send_message("send_image_mode", {}); fig.send_message("refresh", {}); } this.imageObj.onload = function() { if (fig.image_mode == 'full') { // Full images could contain transparency (where diff images // almost always do), so we need to clear the canvas so that // there is no ghosting. fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height); } fig.context.drawImage(fig.imageObj, 0, 0); }; this.imageObj.onunload = function() { this.ws.close(); } this.ws.onmessage = this._make_on_message_function(this); this.ondownload = ondownload; } mpl.figure.prototype._init_header = function() { var titlebar = $( '<div class="ui-dialog-titlebar ui-widget-header ui-corner-all ' + 'ui-helper-clearfix"/>'); var titletext = $( '<div class="ui-dialog-title" style="width: 100%; ' + 'text-align: center; padding: 3px;"/>'); titlebar.append(titletext) this.root.append(titlebar); this.header = titletext[0]; } mpl.figure.prototype._canvas_extra_style = function(canvas_div) { } mpl.figure.prototype._root_extra_style = function(canvas_div) { } mpl.figure.prototype._init_canvas = function() { var fig = this; var canvas_div = $('<div/>'); canvas_div.attr('style', 'position: relative; clear: both; outline: 0'); function canvas_keyboard_event(event) { return fig.key_event(event, event['data']); } canvas_div.keydown('key_press', canvas_keyboard_event); canvas_div.keyup('key_release', canvas_keyboard_event); this.canvas_div = canvas_div this._canvas_extra_style(canvas_div) this.root.append(canvas_div); var canvas = $('<canvas/>'); canvas.addClass('mpl-canvas'); canvas.attr('style', "left: 0; top: 0; z-index: 0; outline: 0") this.canvas = canvas[0]; this.context = canvas[0].getContext("2d"); var rubberband = $('<canvas/>'); rubberband.attr('style', "position: absolute; left: 0; top: 0; z-index: 1;") var pass_mouse_events = true; canvas_div.resizable({ start: function(event, ui) { pass_mouse_events = false; }, resize: function(event, ui) { fig.request_resize(ui.size.width, ui.size.height); }, stop: function(event, ui) { pass_mouse_events = true; fig.request_resize(ui.size.width, ui.size.height); }, }); function mouse_event_fn(event) { if (pass_mouse_events) return fig.mouse_event(event, event['data']); } rubberband.mousedown('button_press', mouse_event_fn); rubberband.mouseup('button_release', mouse_event_fn); // Throttle sequential mouse events to 1 every 20ms. rubberband.mousemove('motion_notify', mouse_event_fn); rubberband.mouseenter('figure_enter', mouse_event_fn); rubberband.mouseleave('figure_leave', mouse_event_fn); canvas_div.on("wheel", function (event) { event = event.originalEvent; event['data'] = 'scroll' if (event.deltaY < 0) { event.step = 1; } else { event.step = -1; } mouse_event_fn(event); }); canvas_div.append(canvas); canvas_div.append(rubberband); this.rubberband = rubberband; this.rubberband_canvas = rubberband[0]; this.rubberband_context = rubberband[0].getContext("2d"); this.rubberband_context.strokeStyle = "#000000"; this._resize_canvas = function(width, height) { // Keep the size of the canvas, canvas container, and rubber band // canvas in synch. canvas_div.css('width', width) canvas_div.css('height', height) canvas.attr('width', width); canvas.attr('height', height); rubberband.attr('width', width); rubberband.attr('height', height); } // Set the figure to an initial 600x600px, this will subsequently be updated // upon first draw. this._resize_canvas(600, 600); // Disable right mouse context menu. $(this.rubberband_canvas).bind("contextmenu",function(e){ return false; }); function set_focus () { canvas.focus(); canvas_div.focus(); } window.setTimeout(set_focus, 100); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items) { var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { // put a spacer in here. continue; } var button = $('<button/>'); button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' + 'ui-button-icon-only'); button.attr('role', 'button'); button.attr('aria-disabled', 'false'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); var icon_img = $('<span/>'); icon_img.addClass('ui-button-icon-primary ui-icon'); icon_img.addClass(image); icon_img.addClass('ui-corner-all'); var tooltip_span = $('<span/>'); tooltip_span.addClass('ui-button-text'); tooltip_span.html(tooltip); button.append(icon_img); button.append(tooltip_span); nav_element.append(button); } var fmt_picker_span = $('<span/>'); var fmt_picker = $('<select/>'); fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content'); fmt_picker_span.append(fmt_picker); nav_element.append(fmt_picker_span); this.format_dropdown = fmt_picker[0]; for (var ind in mpl.extensions) { var fmt = mpl.extensions[ind]; var option = $( '<option/>', {selected: fmt === mpl.default_extension}).html(fmt); fmt_picker.append(option) } // Add hover states to the ui-buttons $( ".ui-button" ).hover( function() { $(this).addClass("ui-state-hover");}, function() { $(this).removeClass("ui-state-hover");} ); var status_bar = $('<span class="mpl-message"/>'); nav_element.append(status_bar); this.message = status_bar[0]; } mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) { // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client, // which will in turn request a refresh of the image. this.send_message('resize', {'width': x_pixels, 'height': y_pixels}); } mpl.figure.prototype.send_message = function(type, properties) { properties['type'] = type; properties['figure_id'] = this.id; this.ws.send(JSON.stringify(properties)); } mpl.figure.prototype.send_draw_message = function() { if (!this.waiting) { this.waiting = true; this.ws.send(JSON.stringify({type: "draw", figure_id: this.id})); } } mpl.figure.prototype.handle_save = function(fig, msg) { var format_dropdown = fig.format_dropdown; var format = format_dropdown.options[format_dropdown.selectedIndex].value; fig.ondownload(fig, format); } mpl.figure.prototype.handle_resize = function(fig, msg) { var size = msg['size']; if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) { fig._resize_canvas(size[0], size[1]); fig.send_message("refresh", {}); }; } mpl.figure.prototype.handle_rubberband = function(fig, msg) { var x0 = msg['x0']; var y0 = fig.canvas.height - msg['y0']; var x1 = msg['x1']; var y1 = fig.canvas.height - msg['y1']; x0 = Math.floor(x0) + 0.5; y0 = Math.floor(y0) + 0.5; x1 = Math.floor(x1) + 0.5; y1 = Math.floor(y1) + 0.5; var min_x = Math.min(x0, x1); var min_y = Math.min(y0, y1); var width = Math.abs(x1 - x0); var height = Math.abs(y1 - y0); fig.rubberband_context.clearRect( 0, 0, fig.canvas.width, fig.canvas.height); fig.rubberband_context.strokeRect(min_x, min_y, width, height); } mpl.figure.prototype.handle_figure_label = function(fig, msg) { // Updates the figure title. fig.header.textContent = msg['label']; } mpl.figure.prototype.handle_cursor = function(fig, msg) { var cursor = msg['cursor']; switch(cursor) { case 0: cursor = 'pointer'; break; case 1: cursor = 'default'; break; case 2: cursor = 'crosshair'; break; case 3: cursor = 'move'; break; } fig.rubberband_canvas.style.cursor = cursor; } mpl.figure.prototype.handle_message = function(fig, msg) { fig.message.textContent = msg['message']; } mpl.figure.prototype.handle_draw = function(fig, msg) { // Request the server to send over a new figure. fig.send_draw_message(); } mpl.figure.prototype.handle_image_mode = function(fig, msg) { fig.image_mode = msg['mode']; } mpl.figure.prototype.updated_canvas_event = function() { // Called whenever the canvas gets updated. this.send_message("ack", {}); } // A function to construct a web socket function for onmessage handling. // Called in the figure constructor. mpl.figure.prototype._make_on_message_function = function(fig) { return function socket_on_message(evt) { if (evt.data instanceof Blob) { /* FIXME: We get "Resource interpreted as Image but * transferred with MIME type text/plain:" errors on * Chrome. But how to set the MIME type? It doesn't seem * to be part of the websocket stream */ evt.data.type = "image/png"; /* Free the memory for the previous frames */ if (fig.imageObj.src) { (window.URL || window.webkitURL).revokeObjectURL( fig.imageObj.src); } fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL( evt.data); fig.updated_canvas_event(); fig.waiting = false; return; } else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == "data:image/png;base64") { fig.imageObj.src = evt.data; fig.updated_canvas_event(); fig.waiting = false; return; } var msg = JSON.parse(evt.data); var msg_type = msg['type']; // Call the "handle_{type}" callback, which takes // the figure and JSON message as its only arguments. try { var callback = fig["handle_" + msg_type]; } catch (e) { console.log("No handler for the '" + msg_type + "' message type: ", msg); return; } if (callback) { try { // console.log("Handling '" + msg_type + "' message: ", msg); callback(fig, msg); } catch (e) { console.log("Exception inside the 'handler_" + msg_type + "' callback:", e, e.stack, msg); } } }; } // from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas mpl.findpos = function(e) { //this section is from http://www.quirksmode.org/js/events_properties.html var targ; if (!e) e = window.event; if (e.target) targ = e.target; else if (e.srcElement) targ = e.srcElement; if (targ.nodeType == 3) // defeat Safari bug targ = targ.parentNode; // jQuery normalizes the pageX and pageY // pageX,Y are the mouse positions relative to the document // offset() returns the position of the element relative to the document var x = e.pageX - $(targ).offset().left; var y = e.pageY - $(targ).offset().top; return {"x": x, "y": y}; }; /* * return a copy of an object with only non-object keys * we need this to avoid circular references * http://stackoverflow.com/a/24161582/3208463 */ function simpleKeys (original) { return Object.keys(original).reduce(function (obj, key) { if (typeof original[key] !== 'object') obj[key] = original[key] return obj; }, {}); } mpl.figure.prototype.mouse_event = function(event, name) { var canvas_pos = mpl.findpos(event) if (name === 'button_press') { this.canvas.focus(); this.canvas_div.focus(); } var x = canvas_pos.x; var y = canvas_pos.y; this.send_message(name, {x: x, y: y, button: event.button, step: event.step, guiEvent: simpleKeys(event)}); /* This prevents the web browser from automatically changing to * the text insertion cursor when the button is pressed. We want * to control all of the cursor setting manually through the * 'cursor' event from matplotlib */ event.preventDefault(); return false; } mpl.figure.prototype._key_event_extra = function(event, name) { // Handle any extra behaviour associated with a key event } mpl.figure.prototype.key_event = function(event, name) { // Prevent repeat events if (name == 'key_press') { if (event.which === this._key) return; else this._key = event.which; } if (name == 'key_release') this._key = null; var value = ''; if (event.ctrlKey && event.which != 17) value += "ctrl+"; if (event.altKey && event.which != 18) value += "alt+"; if (event.shiftKey && event.which != 16) value += "shift+"; value += 'k'; value += event.which.toString(); this._key_event_extra(event, name); this.send_message(name, {key: value, guiEvent: simpleKeys(event)}); return false; } mpl.figure.prototype.toolbar_button_onclick = function(name) { if (name == 'download') { this.handle_save(this, null); } else { this.send_message("toolbar_button", {name: name}); } }; mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) { this.message.textContent = tooltip; }; mpl.toolbar_items = [["Home", "Reset original view", "fa fa-home icon-home", "home"], ["Back", "Back to previous view", "fa fa-arrow-left icon-arrow-left", "back"], ["Forward", "Forward to next view", "fa fa-arrow-right icon-arrow-right", "forward"], ["", "", "", ""], ["Pan", "Pan axes with left mouse, zoom with right", "fa fa-arrows icon-move", "pan"], ["Zoom", "Zoom to rectangle", "fa fa-square-o icon-check-empty", "zoom"], ["", "", "", ""], ["Download", "Download plot", "fa fa-floppy-o icon-save", "download"]]; mpl.extensions = ["eps", "pdf", "png", "ps", "raw", "svg"]; mpl.default_extension = "png";var comm_websocket_adapter = function(comm) { // Create a "websocket"-like object which calls the given IPython comm // object with the appropriate methods. Currently this is a non binary // socket, so there is still some room for performance tuning. var ws = {}; ws.close = function() { comm.close() }; ws.send = function(m) { //console.log('sending', m); comm.send(m); }; // Register the callback with on_msg. comm.on_msg(function(msg) { //console.log('receiving', msg['content']['data'], msg); // Pass the mpl event to the overriden (by mpl) onmessage function. ws.onmessage(msg['content']['data']) }); return ws; } mpl.mpl_figure_comm = function(comm, msg) { // This is the function which gets called when the mpl process // starts-up an IPython Comm through the "matplotlib" channel. var id = msg.content.data.id; // Get hold of the div created by the display call when the Comm // socket was opened in Python. var element = $("#" + id); var ws_proxy = comm_websocket_adapter(comm) function ondownload(figure, format) { window.open(figure.imageObj.src); } var fig = new mpl.figure(id, ws_proxy, ondownload, element.get(0)); // Call onopen now - mpl needs it, as it is assuming we've passed it a real // web socket which is closed, not our websocket->open comm proxy. ws_proxy.onopen(); fig.parent_element = element.get(0); fig.cell_info = mpl.find_output_cell("<div id='" + id + "'></div>"); if (!fig.cell_info) { console.error("Failed to find cell for figure", id, fig); return; } var output_index = fig.cell_info[2] var cell = fig.cell_info[0]; }; mpl.figure.prototype.handle_close = function(fig, msg) { fig.root.unbind('remove') // Update the output cell to use the data from the current canvas. fig.push_to_output(); var dataURL = fig.canvas.toDataURL(); // Re-enable the keyboard manager in IPython - without this line, in FF, // the notebook keyboard shortcuts fail. IPython.keyboard_manager.enable() $(fig.parent_element).html('<img src="' + dataURL + '">'); fig.close_ws(fig, msg); } mpl.figure.prototype.close_ws = function(fig, msg){ fig.send_message('closing', msg); // fig.ws.close() } mpl.figure.prototype.push_to_output = function(remove_interactive) { // Turn the data on the canvas into data in the output cell. var dataURL = this.canvas.toDataURL(); this.cell_info[1]['text/html'] = '<img src="' + dataURL + '">'; } mpl.figure.prototype.updated_canvas_event = function() { // Tell IPython that the notebook contents must change. IPython.notebook.set_dirty(true); this.send_message("ack", {}); var fig = this; // Wait a second, then push the new image to the DOM so // that it is saved nicely (might be nice to debounce this). setTimeout(function () { fig.push_to_output() }, 1000); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items){ var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { continue; }; var button = $('<button class="btn btn-default" href="#" title="' + name + '"><i class="fa ' + image + ' fa-lg"></i></button>'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); nav_element.append(button); } // Add the status bar. var status_bar = $('<span class="mpl-message" style="text-align:right; float: right;"/>'); nav_element.append(status_bar); this.message = status_bar[0]; // Add the close button to the window. var buttongrp = $('<div class="btn-group inline pull-right"></div>'); var button = $('<button class="btn btn-mini btn-primary" href="#" title="Stop Interaction"><i class="fa fa-power-off icon-remove icon-large"></i></button>'); button.click(function (evt) { fig.handle_close(fig, {}); } ); button.mouseover('Stop Interaction', toolbar_mouse_event); buttongrp.append(button); var titlebar = this.root.find($('.ui-dialog-titlebar')); titlebar.prepend(buttongrp); } mpl.figure.prototype._root_extra_style = function(el){ var fig = this el.on("remove", function(){ fig.close_ws(fig, {}); }); } mpl.figure.prototype._canvas_extra_style = function(el){ // this is important to make the div 'focusable el.attr('tabindex', 0) // reach out to IPython and tell the keyboard manager to turn it's self // off when our div gets focus // location in version 3 if (IPython.notebook.keyboard_manager) { IPython.notebook.keyboard_manager.register_events(el); } else { // location in version 2 IPython.keyboard_manager.register_events(el); } } mpl.figure.prototype._key_event_extra = function(event, name) { var manager = IPython.notebook.keyboard_manager; if (!manager) manager = IPython.keyboard_manager; // Check for shift+enter if (event.shiftKey && event.which == 13) { this.canvas_div.blur(); event.shiftKey = false; // Send a "J" for go to next cell event.which = 74; event.keyCode = 74; manager.command_mode(); manager.handle_keydown(event); } } mpl.figure.prototype.handle_save = function(fig, msg) { fig.ondownload(fig, null); } mpl.find_output_cell = function(html_output) { // Return the cell and output element which can be found *uniquely* in the notebook. // Note - this is a bit hacky, but it is done because the "notebook_saving.Notebook" // IPython event is triggered only after the cells have been serialised, which for // our purposes (turning an active figure into a static one), is too late. var cells = IPython.notebook.get_cells(); var ncells = cells.length; for (var i=0; i<ncells; i++) { var cell = cells[i]; if (cell.cell_type === 'code'){ for (var j=0; j<cell.output_area.outputs.length; j++) { var data = cell.output_area.outputs[j]; if (data.data) { // IPython >= 3 moved mimebundle to data attribute of output data = data.data; } if (data['text/html'] == html_output) { return [cell, data, j]; } } } } } // Register the function which deals with the matplotlib target/channel. // The kernel may be null if the page has been refreshed. if (IPython.notebook.kernel != null) { IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm); } </script> </div> </div> <div class="output_area"><div class="prompt"></div> <div class="output_html rendered_html output_subarea "> <img src=""> </div> </div> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[9]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="n">season_ratings</span><span class="p">(</span><span class="s1">&#39;Community&#39;</span><span class="p">)</span> </pre></div> </div> </div> </div> <div class="output_wrapper"> <div class="output"> <div class="output_area"><div class="prompt"></div> <div class="output_subarea output_javascript "> <script type="text/javascript"> /* Put everything inside the global mpl namespace */ window.mpl = {}; mpl.get_websocket_type = function() { if (typeof(WebSocket) !== 'undefined') { return WebSocket; } else if (typeof(MozWebSocket) !== 'undefined') { return MozWebSocket; } else { alert('Your browser does not have WebSocket support.' + 'Please try Chrome, Safari or Firefox ≥ 6. ' + 'Firefox 4 and 5 are also supported but you ' + 'have to enable WebSockets in about:config.'); }; } mpl.figure = function(figure_id, websocket, ondownload, parent_element) { this.id = figure_id; this.ws = websocket; this.supports_binary = (this.ws.binaryType != undefined); if (!this.supports_binary) { var warnings = document.getElementById("mpl-warnings"); if (warnings) { warnings.style.display = 'block'; warnings.textContent = ( "This browser does not support binary websocket messages. " + "Performance may be slow."); } } this.imageObj = new Image(); this.context = undefined; this.message = undefined; this.canvas = undefined; this.rubberband_canvas = undefined; this.rubberband_context = undefined; this.format_dropdown = undefined; this.image_mode = 'full'; this.root = $('<div/>'); this._root_extra_style(this.root) this.root.attr('style', 'display: inline-block'); $(parent_element).append(this.root); this._init_header(this); this._init_canvas(this); this._init_toolbar(this); var fig = this; this.waiting = false; this.ws.onopen = function () { fig.send_message("supports_binary", {value: fig.supports_binary}); fig.send_message("send_image_mode", {}); fig.send_message("refresh", {}); } this.imageObj.onload = function() { if (fig.image_mode == 'full') { // Full images could contain transparency (where diff images // almost always do), so we need to clear the canvas so that // there is no ghosting. fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height); } fig.context.drawImage(fig.imageObj, 0, 0); }; this.imageObj.onunload = function() { this.ws.close(); } this.ws.onmessage = this._make_on_message_function(this); this.ondownload = ondownload; } mpl.figure.prototype._init_header = function() { var titlebar = $( '<div class="ui-dialog-titlebar ui-widget-header ui-corner-all ' + 'ui-helper-clearfix"/>'); var titletext = $( '<div class="ui-dialog-title" style="width: 100%; ' + 'text-align: center; padding: 3px;"/>'); titlebar.append(titletext) this.root.append(titlebar); this.header = titletext[0]; } mpl.figure.prototype._canvas_extra_style = function(canvas_div) { } mpl.figure.prototype._root_extra_style = function(canvas_div) { } mpl.figure.prototype._init_canvas = function() { var fig = this; var canvas_div = $('<div/>'); canvas_div.attr('style', 'position: relative; clear: both; outline: 0'); function canvas_keyboard_event(event) { return fig.key_event(event, event['data']); } canvas_div.keydown('key_press', canvas_keyboard_event); canvas_div.keyup('key_release', canvas_keyboard_event); this.canvas_div = canvas_div this._canvas_extra_style(canvas_div) this.root.append(canvas_div); var canvas = $('<canvas/>'); canvas.addClass('mpl-canvas'); canvas.attr('style', "left: 0; top: 0; z-index: 0; outline: 0") this.canvas = canvas[0]; this.context = canvas[0].getContext("2d"); var rubberband = $('<canvas/>'); rubberband.attr('style', "position: absolute; left: 0; top: 0; z-index: 1;") var pass_mouse_events = true; canvas_div.resizable({ start: function(event, ui) { pass_mouse_events = false; }, resize: function(event, ui) { fig.request_resize(ui.size.width, ui.size.height); }, stop: function(event, ui) { pass_mouse_events = true; fig.request_resize(ui.size.width, ui.size.height); }, }); function mouse_event_fn(event) { if (pass_mouse_events) return fig.mouse_event(event, event['data']); } rubberband.mousedown('button_press', mouse_event_fn); rubberband.mouseup('button_release', mouse_event_fn); // Throttle sequential mouse events to 1 every 20ms. rubberband.mousemove('motion_notify', mouse_event_fn); rubberband.mouseenter('figure_enter', mouse_event_fn); rubberband.mouseleave('figure_leave', mouse_event_fn); canvas_div.on("wheel", function (event) { event = event.originalEvent; event['data'] = 'scroll' if (event.deltaY < 0) { event.step = 1; } else { event.step = -1; } mouse_event_fn(event); }); canvas_div.append(canvas); canvas_div.append(rubberband); this.rubberband = rubberband; this.rubberband_canvas = rubberband[0]; this.rubberband_context = rubberband[0].getContext("2d"); this.rubberband_context.strokeStyle = "#000000"; this._resize_canvas = function(width, height) { // Keep the size of the canvas, canvas container, and rubber band // canvas in synch. canvas_div.css('width', width) canvas_div.css('height', height) canvas.attr('width', width); canvas.attr('height', height); rubberband.attr('width', width); rubberband.attr('height', height); } // Set the figure to an initial 600x600px, this will subsequently be updated // upon first draw. this._resize_canvas(600, 600); // Disable right mouse context menu. $(this.rubberband_canvas).bind("contextmenu",function(e){ return false; }); function set_focus () { canvas.focus(); canvas_div.focus(); } window.setTimeout(set_focus, 100); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items) { var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { // put a spacer in here. continue; } var button = $('<button/>'); button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' + 'ui-button-icon-only'); button.attr('role', 'button'); button.attr('aria-disabled', 'false'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); var icon_img = $('<span/>'); icon_img.addClass('ui-button-icon-primary ui-icon'); icon_img.addClass(image); icon_img.addClass('ui-corner-all'); var tooltip_span = $('<span/>'); tooltip_span.addClass('ui-button-text'); tooltip_span.html(tooltip); button.append(icon_img); button.append(tooltip_span); nav_element.append(button); } var fmt_picker_span = $('<span/>'); var fmt_picker = $('<select/>'); fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content'); fmt_picker_span.append(fmt_picker); nav_element.append(fmt_picker_span); this.format_dropdown = fmt_picker[0]; for (var ind in mpl.extensions) { var fmt = mpl.extensions[ind]; var option = $( '<option/>', {selected: fmt === mpl.default_extension}).html(fmt); fmt_picker.append(option) } // Add hover states to the ui-buttons $( ".ui-button" ).hover( function() { $(this).addClass("ui-state-hover");}, function() { $(this).removeClass("ui-state-hover");} ); var status_bar = $('<span class="mpl-message"/>'); nav_element.append(status_bar); this.message = status_bar[0]; } mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) { // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client, // which will in turn request a refresh of the image. this.send_message('resize', {'width': x_pixels, 'height': y_pixels}); } mpl.figure.prototype.send_message = function(type, properties) { properties['type'] = type; properties['figure_id'] = this.id; this.ws.send(JSON.stringify(properties)); } mpl.figure.prototype.send_draw_message = function() { if (!this.waiting) { this.waiting = true; this.ws.send(JSON.stringify({type: "draw", figure_id: this.id})); } } mpl.figure.prototype.handle_save = function(fig, msg) { var format_dropdown = fig.format_dropdown; var format = format_dropdown.options[format_dropdown.selectedIndex].value; fig.ondownload(fig, format); } mpl.figure.prototype.handle_resize = function(fig, msg) { var size = msg['size']; if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) { fig._resize_canvas(size[0], size[1]); fig.send_message("refresh", {}); }; } mpl.figure.prototype.handle_rubberband = function(fig, msg) { var x0 = msg['x0']; var y0 = fig.canvas.height - msg['y0']; var x1 = msg['x1']; var y1 = fig.canvas.height - msg['y1']; x0 = Math.floor(x0) + 0.5; y0 = Math.floor(y0) + 0.5; x1 = Math.floor(x1) + 0.5; y1 = Math.floor(y1) + 0.5; var min_x = Math.min(x0, x1); var min_y = Math.min(y0, y1); var width = Math.abs(x1 - x0); var height = Math.abs(y1 - y0); fig.rubberband_context.clearRect( 0, 0, fig.canvas.width, fig.canvas.height); fig.rubberband_context.strokeRect(min_x, min_y, width, height); } mpl.figure.prototype.handle_figure_label = function(fig, msg) { // Updates the figure title. fig.header.textContent = msg['label']; } mpl.figure.prototype.handle_cursor = function(fig, msg) { var cursor = msg['cursor']; switch(cursor) { case 0: cursor = 'pointer'; break; case 1: cursor = 'default'; break; case 2: cursor = 'crosshair'; break; case 3: cursor = 'move'; break; } fig.rubberband_canvas.style.cursor = cursor; } mpl.figure.prototype.handle_message = function(fig, msg) { fig.message.textContent = msg['message']; } mpl.figure.prototype.handle_draw = function(fig, msg) { // Request the server to send over a new figure. fig.send_draw_message(); } mpl.figure.prototype.handle_image_mode = function(fig, msg) { fig.image_mode = msg['mode']; } mpl.figure.prototype.updated_canvas_event = function() { // Called whenever the canvas gets updated. this.send_message("ack", {}); } // A function to construct a web socket function for onmessage handling. // Called in the figure constructor. mpl.figure.prototype._make_on_message_function = function(fig) { return function socket_on_message(evt) { if (evt.data instanceof Blob) { /* FIXME: We get "Resource interpreted as Image but * transferred with MIME type text/plain:" errors on * Chrome. But how to set the MIME type? It doesn't seem * to be part of the websocket stream */ evt.data.type = "image/png"; /* Free the memory for the previous frames */ if (fig.imageObj.src) { (window.URL || window.webkitURL).revokeObjectURL( fig.imageObj.src); } fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL( evt.data); fig.updated_canvas_event(); fig.waiting = false; return; } else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == "data:image/png;base64") { fig.imageObj.src = evt.data; fig.updated_canvas_event(); fig.waiting = false; return; } var msg = JSON.parse(evt.data); var msg_type = msg['type']; // Call the "handle_{type}" callback, which takes // the figure and JSON message as its only arguments. try { var callback = fig["handle_" + msg_type]; } catch (e) { console.log("No handler for the '" + msg_type + "' message type: ", msg); return; } if (callback) { try { // console.log("Handling '" + msg_type + "' message: ", msg); callback(fig, msg); } catch (e) { console.log("Exception inside the 'handler_" + msg_type + "' callback:", e, e.stack, msg); } } }; } // from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas mpl.findpos = function(e) { //this section is from http://www.quirksmode.org/js/events_properties.html var targ; if (!e) e = window.event; if (e.target) targ = e.target; else if (e.srcElement) targ = e.srcElement; if (targ.nodeType == 3) // defeat Safari bug targ = targ.parentNode; // jQuery normalizes the pageX and pageY // pageX,Y are the mouse positions relative to the document // offset() returns the position of the element relative to the document var x = e.pageX - $(targ).offset().left; var y = e.pageY - $(targ).offset().top; return {"x": x, "y": y}; }; /* * return a copy of an object with only non-object keys * we need this to avoid circular references * http://stackoverflow.com/a/24161582/3208463 */ function simpleKeys (original) { return Object.keys(original).reduce(function (obj, key) { if (typeof original[key] !== 'object') obj[key] = original[key] return obj; }, {}); } mpl.figure.prototype.mouse_event = function(event, name) { var canvas_pos = mpl.findpos(event) if (name === 'button_press') { this.canvas.focus(); this.canvas_div.focus(); } var x = canvas_pos.x; var y = canvas_pos.y; this.send_message(name, {x: x, y: y, button: event.button, step: event.step, guiEvent: simpleKeys(event)}); /* This prevents the web browser from automatically changing to * the text insertion cursor when the button is pressed. We want * to control all of the cursor setting manually through the * 'cursor' event from matplotlib */ event.preventDefault(); return false; } mpl.figure.prototype._key_event_extra = function(event, name) { // Handle any extra behaviour associated with a key event } mpl.figure.prototype.key_event = function(event, name) { // Prevent repeat events if (name == 'key_press') { if (event.which === this._key) return; else this._key = event.which; } if (name == 'key_release') this._key = null; var value = ''; if (event.ctrlKey && event.which != 17) value += "ctrl+"; if (event.altKey && event.which != 18) value += "alt+"; if (event.shiftKey && event.which != 16) value += "shift+"; value += 'k'; value += event.which.toString(); this._key_event_extra(event, name); this.send_message(name, {key: value, guiEvent: simpleKeys(event)}); return false; } mpl.figure.prototype.toolbar_button_onclick = function(name) { if (name == 'download') { this.handle_save(this, null); } else { this.send_message("toolbar_button", {name: name}); } }; mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) { this.message.textContent = tooltip; }; mpl.toolbar_items = [["Home", "Reset original view", "fa fa-home icon-home", "home"], ["Back", "Back to previous view", "fa fa-arrow-left icon-arrow-left", "back"], ["Forward", "Forward to next view", "fa fa-arrow-right icon-arrow-right", "forward"], ["", "", "", ""], ["Pan", "Pan axes with left mouse, zoom with right", "fa fa-arrows icon-move", "pan"], ["Zoom", "Zoom to rectangle", "fa fa-square-o icon-check-empty", "zoom"], ["", "", "", ""], ["Download", "Download plot", "fa fa-floppy-o icon-save", "download"]]; mpl.extensions = ["eps", "pdf", "png", "ps", "raw", "svg"]; mpl.default_extension = "png";var comm_websocket_adapter = function(comm) { // Create a "websocket"-like object which calls the given IPython comm // object with the appropriate methods. Currently this is a non binary // socket, so there is still some room for performance tuning. var ws = {}; ws.close = function() { comm.close() }; ws.send = function(m) { //console.log('sending', m); comm.send(m); }; // Register the callback with on_msg. comm.on_msg(function(msg) { //console.log('receiving', msg['content']['data'], msg); // Pass the mpl event to the overriden (by mpl) onmessage function. ws.onmessage(msg['content']['data']) }); return ws; } mpl.mpl_figure_comm = function(comm, msg) { // This is the function which gets called when the mpl process // starts-up an IPython Comm through the "matplotlib" channel. var id = msg.content.data.id; // Get hold of the div created by the display call when the Comm // socket was opened in Python. var element = $("#" + id); var ws_proxy = comm_websocket_adapter(comm) function ondownload(figure, format) { window.open(figure.imageObj.src); } var fig = new mpl.figure(id, ws_proxy, ondownload, element.get(0)); // Call onopen now - mpl needs it, as it is assuming we've passed it a real // web socket which is closed, not our websocket->open comm proxy. ws_proxy.onopen(); fig.parent_element = element.get(0); fig.cell_info = mpl.find_output_cell("<div id='" + id + "'></div>"); if (!fig.cell_info) { console.error("Failed to find cell for figure", id, fig); return; } var output_index = fig.cell_info[2] var cell = fig.cell_info[0]; }; mpl.figure.prototype.handle_close = function(fig, msg) { fig.root.unbind('remove') // Update the output cell to use the data from the current canvas. fig.push_to_output(); var dataURL = fig.canvas.toDataURL(); // Re-enable the keyboard manager in IPython - without this line, in FF, // the notebook keyboard shortcuts fail. IPython.keyboard_manager.enable() $(fig.parent_element).html('<img src="' + dataURL + '">'); fig.close_ws(fig, msg); } mpl.figure.prototype.close_ws = function(fig, msg){ fig.send_message('closing', msg); // fig.ws.close() } mpl.figure.prototype.push_to_output = function(remove_interactive) { // Turn the data on the canvas into data in the output cell. var dataURL = this.canvas.toDataURL(); this.cell_info[1]['text/html'] = '<img src="' + dataURL + '">'; } mpl.figure.prototype.updated_canvas_event = function() { // Tell IPython that the notebook contents must change. IPython.notebook.set_dirty(true); this.send_message("ack", {}); var fig = this; // Wait a second, then push the new image to the DOM so // that it is saved nicely (might be nice to debounce this). setTimeout(function () { fig.push_to_output() }, 1000); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items){ var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { continue; }; var button = $('<button class="btn btn-default" href="#" title="' + name + '"><i class="fa ' + image + ' fa-lg"></i></button>'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); nav_element.append(button); } // Add the status bar. var status_bar = $('<span class="mpl-message" style="text-align:right; float: right;"/>'); nav_element.append(status_bar); this.message = status_bar[0]; // Add the close button to the window. var buttongrp = $('<div class="btn-group inline pull-right"></div>'); var button = $('<button class="btn btn-mini btn-primary" href="#" title="Stop Interaction"><i class="fa fa-power-off icon-remove icon-large"></i></button>'); button.click(function (evt) { fig.handle_close(fig, {}); } ); button.mouseover('Stop Interaction', toolbar_mouse_event); buttongrp.append(button); var titlebar = this.root.find($('.ui-dialog-titlebar')); titlebar.prepend(buttongrp); } mpl.figure.prototype._root_extra_style = function(el){ var fig = this el.on("remove", function(){ fig.close_ws(fig, {}); }); } mpl.figure.prototype._canvas_extra_style = function(el){ // this is important to make the div 'focusable el.attr('tabindex', 0) // reach out to IPython and tell the keyboard manager to turn it's self // off when our div gets focus // location in version 3 if (IPython.notebook.keyboard_manager) { IPython.notebook.keyboard_manager.register_events(el); } else { // location in version 2 IPython.keyboard_manager.register_events(el); } } mpl.figure.prototype._key_event_extra = function(event, name) { var manager = IPython.notebook.keyboard_manager; if (!manager) manager = IPython.keyboard_manager; // Check for shift+enter if (event.shiftKey && event.which == 13) { this.canvas_div.blur(); event.shiftKey = false; // Send a "J" for go to next cell event.which = 74; event.keyCode = 74; manager.command_mode(); manager.handle_keydown(event); } } mpl.figure.prototype.handle_save = function(fig, msg) { fig.ondownload(fig, null); } mpl.find_output_cell = function(html_output) { // Return the cell and output element which can be found *uniquely* in the notebook. // Note - this is a bit hacky, but it is done because the "notebook_saving.Notebook" // IPython event is triggered only after the cells have been serialised, which for // our purposes (turning an active figure into a static one), is too late. var cells = IPython.notebook.get_cells(); var ncells = cells.length; for (var i=0; i<ncells; i++) { var cell = cells[i]; if (cell.cell_type === 'code'){ for (var j=0; j<cell.output_area.outputs.length; j++) { var data = cell.output_area.outputs[j]; if (data.data) { // IPython >= 3 moved mimebundle to data attribute of output data = data.data; } if (data['text/html'] == html_output) { return [cell, data, j]; } } } } } // Register the function which deals with the matplotlib target/channel. // The kernel may be null if the page has been refreshed. if (IPython.notebook.kernel != null) { IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm); } </script> </div> </div> <div class="output_area"><div class="prompt"></div> <div class="output_html rendered_html output_subarea "> <img src=""> </div> </div> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <h2 id="Ratingverl&#228;ufe-unterschiedlicher-Serien">Ratingverl&#228;ufe unterschiedlicher Serien<a class="anchor-link" href="#Ratingverl&#228;ufe-unterschiedlicher-Serien">&#182;</a></h2> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Nun lassen wir mal ein paar Serien gegeneinander antreten. Dabei betrachten wir immer alle Episoden aller Staffeln.</p> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[12]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="k">def</span> <span class="nf">series_ratings</span><span class="p">(</span><span class="n">series</span><span class="p">):</span> <span class="n">data</span> <span class="o">=</span> <span class="p">{}</span> <span class="k">for</span> <span class="n">serie</span> <span class="ow">in</span> <span class="n">series</span><span class="p">:</span> <span class="n">show</span> <span class="o">=</span> <span class="n">TVShow</span><span class="p">(</span><span class="n">serie</span><span class="p">)</span> <span class="n">ratings</span> <span class="o">=</span> <span class="p">[]</span> <span class="k">for</span> <span class="n">season</span> <span class="ow">in</span> <span class="n">show</span><span class="o">.</span><span class="n">seasons</span><span class="p">:</span> <span class="k">if</span> <span class="n">season</span><span class="o">.</span><span class="n">season</span> <span class="o">==</span> <span class="mi">0</span><span class="p">:</span> <span class="k">continue</span> <span class="k">for</span> <span class="n">episode</span> <span class="ow">in</span> <span class="n">season</span><span class="o">.</span><span class="n">episodes</span><span class="p">:</span> <span class="n">ratings</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="n">episode</span><span class="o">.</span><span class="n">rating</span><span class="p">)</span> <span class="n">data</span><span class="p">[</span><span class="n">serie</span><span class="p">]</span> <span class="o">=</span> <span class="n">pd</span><span class="o">.</span><span class="n">Series</span><span class="p">(</span><span class="n">ratings</span><span class="p">)</span> <span class="n">df</span> <span class="o">=</span> <span class="n">pd</span><span class="o">.</span><span class="n">DataFrame</span><span class="p">(</span><span class="n">data</span><span class="p">)</span> <span class="n">df</span><span class="o">.</span><span class="n">plot</span><span class="p">(</span><span class="n">colormap</span><span class="o">=</span><span class="n">matplotlib</span><span class="o">.</span><span class="n">cm</span><span class="o">.</span><span class="n">Accent</span><span class="p">)</span> </pre></div> </div> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <h3 id="Beispiel">Beispiel<a class="anchor-link" href="#Beispiel">&#182;</a></h3> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[13]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="n">series_ratings</span><span class="p">([</span><span class="s1">&#39;Gotham&#39;</span><span class="p">,</span> <span class="s1">&#39;Flash&#39;</span><span class="p">,</span> <span class="s1">&#39;Arrow&#39;</span><span class="p">,</span> <span class="s1">&#39;Supergirl&#39;</span><span class="p">])</span> </pre></div> </div> </div> </div> <div class="output_wrapper"> <div class="output"> <div class="output_area"><div class="prompt"></div> <div class="output_subarea output_javascript "> <script type="text/javascript"> /* Put everything inside the global mpl namespace */ window.mpl = {}; mpl.get_websocket_type = function() { if (typeof(WebSocket) !== 'undefined') { return WebSocket; } else if (typeof(MozWebSocket) !== 'undefined') { return MozWebSocket; } else { alert('Your browser does not have WebSocket support.' + 'Please try Chrome, Safari or Firefox ≥ 6. ' + 'Firefox 4 and 5 are also supported but you ' + 'have to enable WebSockets in about:config.'); }; } mpl.figure = function(figure_id, websocket, ondownload, parent_element) { this.id = figure_id; this.ws = websocket; this.supports_binary = (this.ws.binaryType != undefined); if (!this.supports_binary) { var warnings = document.getElementById("mpl-warnings"); if (warnings) { warnings.style.display = 'block'; warnings.textContent = ( "This browser does not support binary websocket messages. " + "Performance may be slow."); } } this.imageObj = new Image(); this.context = undefined; this.message = undefined; this.canvas = undefined; this.rubberband_canvas = undefined; this.rubberband_context = undefined; this.format_dropdown = undefined; this.image_mode = 'full'; this.root = $('<div/>'); this._root_extra_style(this.root) this.root.attr('style', 'display: inline-block'); $(parent_element).append(this.root); this._init_header(this); this._init_canvas(this); this._init_toolbar(this); var fig = this; this.waiting = false; this.ws.onopen = function () { fig.send_message("supports_binary", {value: fig.supports_binary}); fig.send_message("send_image_mode", {}); fig.send_message("refresh", {}); } this.imageObj.onload = function() { if (fig.image_mode == 'full') { // Full images could contain transparency (where diff images // almost always do), so we need to clear the canvas so that // there is no ghosting. fig.context.clearRect(0, 0, fig.canvas.width, fig.canvas.height); } fig.context.drawImage(fig.imageObj, 0, 0); }; this.imageObj.onunload = function() { this.ws.close(); } this.ws.onmessage = this._make_on_message_function(this); this.ondownload = ondownload; } mpl.figure.prototype._init_header = function() { var titlebar = $( '<div class="ui-dialog-titlebar ui-widget-header ui-corner-all ' + 'ui-helper-clearfix"/>'); var titletext = $( '<div class="ui-dialog-title" style="width: 100%; ' + 'text-align: center; padding: 3px;"/>'); titlebar.append(titletext) this.root.append(titlebar); this.header = titletext[0]; } mpl.figure.prototype._canvas_extra_style = function(canvas_div) { } mpl.figure.prototype._root_extra_style = function(canvas_div) { } mpl.figure.prototype._init_canvas = function() { var fig = this; var canvas_div = $('<div/>'); canvas_div.attr('style', 'position: relative; clear: both; outline: 0'); function canvas_keyboard_event(event) { return fig.key_event(event, event['data']); } canvas_div.keydown('key_press', canvas_keyboard_event); canvas_div.keyup('key_release', canvas_keyboard_event); this.canvas_div = canvas_div this._canvas_extra_style(canvas_div) this.root.append(canvas_div); var canvas = $('<canvas/>'); canvas.addClass('mpl-canvas'); canvas.attr('style', "left: 0; top: 0; z-index: 0; outline: 0") this.canvas = canvas[0]; this.context = canvas[0].getContext("2d"); var rubberband = $('<canvas/>'); rubberband.attr('style', "position: absolute; left: 0; top: 0; z-index: 1;") var pass_mouse_events = true; canvas_div.resizable({ start: function(event, ui) { pass_mouse_events = false; }, resize: function(event, ui) { fig.request_resize(ui.size.width, ui.size.height); }, stop: function(event, ui) { pass_mouse_events = true; fig.request_resize(ui.size.width, ui.size.height); }, }); function mouse_event_fn(event) { if (pass_mouse_events) return fig.mouse_event(event, event['data']); } rubberband.mousedown('button_press', mouse_event_fn); rubberband.mouseup('button_release', mouse_event_fn); // Throttle sequential mouse events to 1 every 20ms. rubberband.mousemove('motion_notify', mouse_event_fn); rubberband.mouseenter('figure_enter', mouse_event_fn); rubberband.mouseleave('figure_leave', mouse_event_fn); canvas_div.on("wheel", function (event) { event = event.originalEvent; event['data'] = 'scroll' if (event.deltaY < 0) { event.step = 1; } else { event.step = -1; } mouse_event_fn(event); }); canvas_div.append(canvas); canvas_div.append(rubberband); this.rubberband = rubberband; this.rubberband_canvas = rubberband[0]; this.rubberband_context = rubberband[0].getContext("2d"); this.rubberband_context.strokeStyle = "#000000"; this._resize_canvas = function(width, height) { // Keep the size of the canvas, canvas container, and rubber band // canvas in synch. canvas_div.css('width', width) canvas_div.css('height', height) canvas.attr('width', width); canvas.attr('height', height); rubberband.attr('width', width); rubberband.attr('height', height); } // Set the figure to an initial 600x600px, this will subsequently be updated // upon first draw. this._resize_canvas(600, 600); // Disable right mouse context menu. $(this.rubberband_canvas).bind("contextmenu",function(e){ return false; }); function set_focus () { canvas.focus(); canvas_div.focus(); } window.setTimeout(set_focus, 100); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items) { var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { // put a spacer in here. continue; } var button = $('<button/>'); button.addClass('ui-button ui-widget ui-state-default ui-corner-all ' + 'ui-button-icon-only'); button.attr('role', 'button'); button.attr('aria-disabled', 'false'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); var icon_img = $('<span/>'); icon_img.addClass('ui-button-icon-primary ui-icon'); icon_img.addClass(image); icon_img.addClass('ui-corner-all'); var tooltip_span = $('<span/>'); tooltip_span.addClass('ui-button-text'); tooltip_span.html(tooltip); button.append(icon_img); button.append(tooltip_span); nav_element.append(button); } var fmt_picker_span = $('<span/>'); var fmt_picker = $('<select/>'); fmt_picker.addClass('mpl-toolbar-option ui-widget ui-widget-content'); fmt_picker_span.append(fmt_picker); nav_element.append(fmt_picker_span); this.format_dropdown = fmt_picker[0]; for (var ind in mpl.extensions) { var fmt = mpl.extensions[ind]; var option = $( '<option/>', {selected: fmt === mpl.default_extension}).html(fmt); fmt_picker.append(option) } // Add hover states to the ui-buttons $( ".ui-button" ).hover( function() { $(this).addClass("ui-state-hover");}, function() { $(this).removeClass("ui-state-hover");} ); var status_bar = $('<span class="mpl-message"/>'); nav_element.append(status_bar); this.message = status_bar[0]; } mpl.figure.prototype.request_resize = function(x_pixels, y_pixels) { // Request matplotlib to resize the figure. Matplotlib will then trigger a resize in the client, // which will in turn request a refresh of the image. this.send_message('resize', {'width': x_pixels, 'height': y_pixels}); } mpl.figure.prototype.send_message = function(type, properties) { properties['type'] = type; properties['figure_id'] = this.id; this.ws.send(JSON.stringify(properties)); } mpl.figure.prototype.send_draw_message = function() { if (!this.waiting) { this.waiting = true; this.ws.send(JSON.stringify({type: "draw", figure_id: this.id})); } } mpl.figure.prototype.handle_save = function(fig, msg) { var format_dropdown = fig.format_dropdown; var format = format_dropdown.options[format_dropdown.selectedIndex].value; fig.ondownload(fig, format); } mpl.figure.prototype.handle_resize = function(fig, msg) { var size = msg['size']; if (size[0] != fig.canvas.width || size[1] != fig.canvas.height) { fig._resize_canvas(size[0], size[1]); fig.send_message("refresh", {}); }; } mpl.figure.prototype.handle_rubberband = function(fig, msg) { var x0 = msg['x0']; var y0 = fig.canvas.height - msg['y0']; var x1 = msg['x1']; var y1 = fig.canvas.height - msg['y1']; x0 = Math.floor(x0) + 0.5; y0 = Math.floor(y0) + 0.5; x1 = Math.floor(x1) + 0.5; y1 = Math.floor(y1) + 0.5; var min_x = Math.min(x0, x1); var min_y = Math.min(y0, y1); var width = Math.abs(x1 - x0); var height = Math.abs(y1 - y0); fig.rubberband_context.clearRect( 0, 0, fig.canvas.width, fig.canvas.height); fig.rubberband_context.strokeRect(min_x, min_y, width, height); } mpl.figure.prototype.handle_figure_label = function(fig, msg) { // Updates the figure title. fig.header.textContent = msg['label']; } mpl.figure.prototype.handle_cursor = function(fig, msg) { var cursor = msg['cursor']; switch(cursor) { case 0: cursor = 'pointer'; break; case 1: cursor = 'default'; break; case 2: cursor = 'crosshair'; break; case 3: cursor = 'move'; break; } fig.rubberband_canvas.style.cursor = cursor; } mpl.figure.prototype.handle_message = function(fig, msg) { fig.message.textContent = msg['message']; } mpl.figure.prototype.handle_draw = function(fig, msg) { // Request the server to send over a new figure. fig.send_draw_message(); } mpl.figure.prototype.handle_image_mode = function(fig, msg) { fig.image_mode = msg['mode']; } mpl.figure.prototype.updated_canvas_event = function() { // Called whenever the canvas gets updated. this.send_message("ack", {}); } // A function to construct a web socket function for onmessage handling. // Called in the figure constructor. mpl.figure.prototype._make_on_message_function = function(fig) { return function socket_on_message(evt) { if (evt.data instanceof Blob) { /* FIXME: We get "Resource interpreted as Image but * transferred with MIME type text/plain:" errors on * Chrome. But how to set the MIME type? It doesn't seem * to be part of the websocket stream */ evt.data.type = "image/png"; /* Free the memory for the previous frames */ if (fig.imageObj.src) { (window.URL || window.webkitURL).revokeObjectURL( fig.imageObj.src); } fig.imageObj.src = (window.URL || window.webkitURL).createObjectURL( evt.data); fig.updated_canvas_event(); fig.waiting = false; return; } else if (typeof evt.data === 'string' && evt.data.slice(0, 21) == "data:image/png;base64") { fig.imageObj.src = evt.data; fig.updated_canvas_event(); fig.waiting = false; return; } var msg = JSON.parse(evt.data); var msg_type = msg['type']; // Call the "handle_{type}" callback, which takes // the figure and JSON message as its only arguments. try { var callback = fig["handle_" + msg_type]; } catch (e) { console.log("No handler for the '" + msg_type + "' message type: ", msg); return; } if (callback) { try { // console.log("Handling '" + msg_type + "' message: ", msg); callback(fig, msg); } catch (e) { console.log("Exception inside the 'handler_" + msg_type + "' callback:", e, e.stack, msg); } } }; } // from http://stackoverflow.com/questions/1114465/getting-mouse-location-in-canvas mpl.findpos = function(e) { //this section is from http://www.quirksmode.org/js/events_properties.html var targ; if (!e) e = window.event; if (e.target) targ = e.target; else if (e.srcElement) targ = e.srcElement; if (targ.nodeType == 3) // defeat Safari bug targ = targ.parentNode; // jQuery normalizes the pageX and pageY // pageX,Y are the mouse positions relative to the document // offset() returns the position of the element relative to the document var x = e.pageX - $(targ).offset().left; var y = e.pageY - $(targ).offset().top; return {"x": x, "y": y}; }; /* * return a copy of an object with only non-object keys * we need this to avoid circular references * http://stackoverflow.com/a/24161582/3208463 */ function simpleKeys (original) { return Object.keys(original).reduce(function (obj, key) { if (typeof original[key] !== 'object') obj[key] = original[key] return obj; }, {}); } mpl.figure.prototype.mouse_event = function(event, name) { var canvas_pos = mpl.findpos(event) if (name === 'button_press') { this.canvas.focus(); this.canvas_div.focus(); } var x = canvas_pos.x; var y = canvas_pos.y; this.send_message(name, {x: x, y: y, button: event.button, step: event.step, guiEvent: simpleKeys(event)}); /* This prevents the web browser from automatically changing to * the text insertion cursor when the button is pressed. We want * to control all of the cursor setting manually through the * 'cursor' event from matplotlib */ event.preventDefault(); return false; } mpl.figure.prototype._key_event_extra = function(event, name) { // Handle any extra behaviour associated with a key event } mpl.figure.prototype.key_event = function(event, name) { // Prevent repeat events if (name == 'key_press') { if (event.which === this._key) return; else this._key = event.which; } if (name == 'key_release') this._key = null; var value = ''; if (event.ctrlKey && event.which != 17) value += "ctrl+"; if (event.altKey && event.which != 18) value += "alt+"; if (event.shiftKey && event.which != 16) value += "shift+"; value += 'k'; value += event.which.toString(); this._key_event_extra(event, name); this.send_message(name, {key: value, guiEvent: simpleKeys(event)}); return false; } mpl.figure.prototype.toolbar_button_onclick = function(name) { if (name == 'download') { this.handle_save(this, null); } else { this.send_message("toolbar_button", {name: name}); } }; mpl.figure.prototype.toolbar_button_onmouseover = function(tooltip) { this.message.textContent = tooltip; }; mpl.toolbar_items = [["Home", "Reset original view", "fa fa-home icon-home", "home"], ["Back", "Back to previous view", "fa fa-arrow-left icon-arrow-left", "back"], ["Forward", "Forward to next view", "fa fa-arrow-right icon-arrow-right", "forward"], ["", "", "", ""], ["Pan", "Pan axes with left mouse, zoom with right", "fa fa-arrows icon-move", "pan"], ["Zoom", "Zoom to rectangle", "fa fa-square-o icon-check-empty", "zoom"], ["", "", "", ""], ["Download", "Download plot", "fa fa-floppy-o icon-save", "download"]]; mpl.extensions = ["eps", "pdf", "png", "ps", "raw", "svg"]; mpl.default_extension = "png";var comm_websocket_adapter = function(comm) { // Create a "websocket"-like object which calls the given IPython comm // object with the appropriate methods. Currently this is a non binary // socket, so there is still some room for performance tuning. var ws = {}; ws.close = function() { comm.close() }; ws.send = function(m) { //console.log('sending', m); comm.send(m); }; // Register the callback with on_msg. comm.on_msg(function(msg) { //console.log('receiving', msg['content']['data'], msg); // Pass the mpl event to the overriden (by mpl) onmessage function. ws.onmessage(msg['content']['data']) }); return ws; } mpl.mpl_figure_comm = function(comm, msg) { // This is the function which gets called when the mpl process // starts-up an IPython Comm through the "matplotlib" channel. var id = msg.content.data.id; // Get hold of the div created by the display call when the Comm // socket was opened in Python. var element = $("#" + id); var ws_proxy = comm_websocket_adapter(comm) function ondownload(figure, format) { window.open(figure.imageObj.src); } var fig = new mpl.figure(id, ws_proxy, ondownload, element.get(0)); // Call onopen now - mpl needs it, as it is assuming we've passed it a real // web socket which is closed, not our websocket->open comm proxy. ws_proxy.onopen(); fig.parent_element = element.get(0); fig.cell_info = mpl.find_output_cell("<div id='" + id + "'></div>"); if (!fig.cell_info) { console.error("Failed to find cell for figure", id, fig); return; } var output_index = fig.cell_info[2] var cell = fig.cell_info[0]; }; mpl.figure.prototype.handle_close = function(fig, msg) { fig.root.unbind('remove') // Update the output cell to use the data from the current canvas. fig.push_to_output(); var dataURL = fig.canvas.toDataURL(); // Re-enable the keyboard manager in IPython - without this line, in FF, // the notebook keyboard shortcuts fail. IPython.keyboard_manager.enable() $(fig.parent_element).html('<img src="' + dataURL + '">'); fig.close_ws(fig, msg); } mpl.figure.prototype.close_ws = function(fig, msg){ fig.send_message('closing', msg); // fig.ws.close() } mpl.figure.prototype.push_to_output = function(remove_interactive) { // Turn the data on the canvas into data in the output cell. var dataURL = this.canvas.toDataURL(); this.cell_info[1]['text/html'] = '<img src="' + dataURL + '">'; } mpl.figure.prototype.updated_canvas_event = function() { // Tell IPython that the notebook contents must change. IPython.notebook.set_dirty(true); this.send_message("ack", {}); var fig = this; // Wait a second, then push the new image to the DOM so // that it is saved nicely (might be nice to debounce this). setTimeout(function () { fig.push_to_output() }, 1000); } mpl.figure.prototype._init_toolbar = function() { var fig = this; var nav_element = $('<div/>') nav_element.attr('style', 'width: 100%'); this.root.append(nav_element); // Define a callback function for later on. function toolbar_event(event) { return fig.toolbar_button_onclick(event['data']); } function toolbar_mouse_event(event) { return fig.toolbar_button_onmouseover(event['data']); } for(var toolbar_ind in mpl.toolbar_items){ var name = mpl.toolbar_items[toolbar_ind][0]; var tooltip = mpl.toolbar_items[toolbar_ind][1]; var image = mpl.toolbar_items[toolbar_ind][2]; var method_name = mpl.toolbar_items[toolbar_ind][3]; if (!name) { continue; }; var button = $('<button class="btn btn-default" href="#" title="' + name + '"><i class="fa ' + image + ' fa-lg"></i></button>'); button.click(method_name, toolbar_event); button.mouseover(tooltip, toolbar_mouse_event); nav_element.append(button); } // Add the status bar. var status_bar = $('<span class="mpl-message" style="text-align:right; float: right;"/>'); nav_element.append(status_bar); this.message = status_bar[0]; // Add the close button to the window. var buttongrp = $('<div class="btn-group inline pull-right"></div>'); var button = $('<button class="btn btn-mini btn-primary" href="#" title="Stop Interaction"><i class="fa fa-power-off icon-remove icon-large"></i></button>'); button.click(function (evt) { fig.handle_close(fig, {}); } ); button.mouseover('Stop Interaction', toolbar_mouse_event); buttongrp.append(button); var titlebar = this.root.find($('.ui-dialog-titlebar')); titlebar.prepend(buttongrp); } mpl.figure.prototype._root_extra_style = function(el){ var fig = this el.on("remove", function(){ fig.close_ws(fig, {}); }); } mpl.figure.prototype._canvas_extra_style = function(el){ // this is important to make the div 'focusable el.attr('tabindex', 0) // reach out to IPython and tell the keyboard manager to turn it's self // off when our div gets focus // location in version 3 if (IPython.notebook.keyboard_manager) { IPython.notebook.keyboard_manager.register_events(el); } else { // location in version 2 IPython.keyboard_manager.register_events(el); } } mpl.figure.prototype._key_event_extra = function(event, name) { var manager = IPython.notebook.keyboard_manager; if (!manager) manager = IPython.keyboard_manager; // Check for shift+enter if (event.shiftKey && event.which == 13) { this.canvas_div.blur(); event.shiftKey = false; // Send a "J" for go to next cell event.which = 74; event.keyCode = 74; manager.command_mode(); manager.handle_keydown(event); } } mpl.figure.prototype.handle_save = function(fig, msg) { fig.ondownload(fig, null); } mpl.find_output_cell = function(html_output) { // Return the cell and output element which can be found *uniquely* in the notebook. // Note - this is a bit hacky, but it is done because the "notebook_saving.Notebook" // IPython event is triggered only after the cells have been serialised, which for // our purposes (turning an active figure into a static one), is too late. var cells = IPython.notebook.get_cells(); var ncells = cells.length; for (var i=0; i<ncells; i++) { var cell = cells[i]; if (cell.cell_type === 'code'){ for (var j=0; j<cell.output_area.outputs.length; j++) { var data = cell.output_area.outputs[j]; if (data.data) { // IPython >= 3 moved mimebundle to data attribute of output data = data.data; } if (data['text/html'] == html_output) { return [cell, data, j]; } } } } } // Register the function which deals with the matplotlib target/channel. // The kernel may be null if the page has been refreshed. if (IPython.notebook.kernel != null) { IPython.notebook.kernel.comm_manager.register_target('matplotlib', mpl.mpl_figure_comm); } </script> </div> </div> <div class="output_area"><div class="prompt"></div> <div class="output_html rendered_html output_subarea "> <img src=""> </div> </div> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Mir ist gerade nichts Besseres eingefallen. Es müssen ja in irgendeinen Sinn vergleichbare Serien sein. Da habe ich mir mal die vier aktuellen DC-Serien angeschaut. Da Arrow schon einige Episoden mehr hat, habe ich den Plot angepasst und hinten abgeschnitten. Glaubt man den Ratings, hätte ich Supergirl doch mehr Chancen als zwei Folgen geben sollen. War in den ersten Folgen einfach zu glatt gebügelt.</p> </div> </div> </div></p>A Film About Coffee2016-02-12T12:51:00+01:002016-02-12T12:51:00+01:00marvintag:xsteadfastx.org,2016-02-12:/2016/02/12/a-film-about-coffee/<p>Auf Vimeo kann man eine zemlich feine Dokumentation über Spezialitätenkaffee und Kaffeekultur <a href="https://vimeo.com/ondemand/afilmaboutcoffee/92370761">kaufen</a>. Ich fand ihn wirklich gut gemacht. Kaffeekultur fasziniert mich. Ich besitze mitlerweile eine Reihe an Kaffeezubereiter und doch werden meine Beine beim <a href="http://www.coffeecircle.com/kaffeezubereitung/kaffee-syphon/">Siphon</a> leicht zittrig. Wie super sieht das aus bitte? Bloß wie kann ich das gegenüber …</p><p>Auf Vimeo kann man eine zemlich feine Dokumentation über Spezialitätenkaffee und Kaffeekultur <a href="https://vimeo.com/ondemand/afilmaboutcoffee/92370761">kaufen</a>. Ich fand ihn wirklich gut gemacht. Kaffeekultur fasziniert mich. Ich besitze mitlerweile eine Reihe an Kaffeezubereiter und doch werden meine Beine beim <a href="http://www.coffeecircle.com/kaffeezubereitung/kaffee-syphon/">Siphon</a> leicht zittrig. Wie super sieht das aus bitte? Bloß wie kann ich das gegenüber meinem Bankkonto rechtfertigen?</p> <blockquote> <p>A Film About Coffee is a love letter to, and meditation on, specialty coffee. It examines what it takes, and what it means, for coffee to be defined as “specialty.” The film whisks audiences on a trip around the world, from farms in Honduras and Rwanda to coffee shops in Tokyo, Portland, Seattle, San Francisco and New York. Through the eyes and experiences of farmers and baristas, the film offers a unique overview of all the elements—the processes, preferences and preparations; traditions old and new—that come together to create the best cups. This is a film that bridges gaps both intellectual and geographical, evoking flavor and pleasure, and providing both as well.</p> </blockquote> <p><span class="videobox"> <iframe src="//player.vimeo.com/video/92370761?title=0&amp;byline=0&amp;portrait=0" width="640" height="390" frameborder="0" webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Pity Sex - Burden You2016-02-11T10:45:00+01:002016-02-11T10:45:00+01:00marvintag:xsteadfastx.org,2016-02-11:/2016/02/11/pity-sex-burden-you/<p>Hier mal ein neuer Track von Pity Sex. Das neue Album kommt im April. Wird auch Zeit.</p> <p><iframe width="100%" height="400" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?visual=true&url=http%3A%2F%2Fapi.soundcloud.com%2Ftracks%2F202234802&show_artwork=true&secret_token=s-nCHvQ"></iframe></p><p>Hier mal ein neuer Track von Pity Sex. Das neue Album kommt im April. Wird auch Zeit.</p> <p><iframe width="100%" height="400" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?visual=true&url=http%3A%2F%2Fapi.soundcloud.com%2Ftracks%2F202234802&show_artwork=true&secret_token=s-nCHvQ"></iframe></p>Hörspiel: Das Hacker-Syndrom2016-02-10T13:24:00+01:002016-02-10T13:24:00+01:00marvintag:xsteadfastx.org,2016-02-10:/2016/02/10/hoerspiel-das-hacker-syndrom/<p><img alt="Telecomix" src="http://xsteadfastx.org/images/telecomix.png"></p> <p>Hier ein kleiner Podcasttipp zwischendurch. <a href="http://www.wdr3.de/programm/sendungen/wdr3hoerspiel/hackersyndrom-100.html">Das Hacker-Syndrom</a> von Johannes Nichelmann. Es befasst sich mit <a href="http://herrurbach.de/">Stephan Urbach</a> und die Zeit des arabischen Frühlings. Das Hörspiel hat mich wirklich sehr berührt.</p> <p><audio controls><source src="http://podcast-ww.wdr.de/medstdp/fsk0/91/916466/wdr3hoerspiel_2016-01-26_dashackersyndromwdr3hoerspiel26012016_wdr3.mp3" type="audio/mpeg">Your browser does not support the audio element.</audio></p><p><img alt="Telecomix" src="http://xsteadfastx.org/images/telecomix.png"></p> <p>Hier ein kleiner Podcasttipp zwischendurch. <a href="http://www.wdr3.de/programm/sendungen/wdr3hoerspiel/hackersyndrom-100.html">Das Hacker-Syndrom</a> von Johannes Nichelmann. Es befasst sich mit <a href="http://herrurbach.de/">Stephan Urbach</a> und die Zeit des arabischen Frühlings. Das Hörspiel hat mich wirklich sehr berührt.</p> <p><audio controls><source src="http://podcast-ww.wdr.de/medstdp/fsk0/91/916466/wdr3hoerspiel_2016-01-26_dashackersyndromwdr3hoerspiel26012016_wdr3.mp3" type="audio/mpeg">Your browser does not support the audio element.</audio></p>Memoryhouse sind wieder da2016-02-10T10:02:00+01:002016-02-10T10:02:00+01:00marvintag:xsteadfastx.org,2016-02-10:/2016/02/10/memoryhouse-sind-wieder-da/<p><img alt="Soft Hate" src="http://xsteadfastx.org/images/memoryhouse-softhate.jpg"></p> <p>Eigentlich dachte ich das es das war mit Dreampop/Chillwave. 5 bis 6 Jahre und man denkt man hat alles gehört. Jedes 90er Zitat bis zum erbrechen gebracht. Wie kann man sich da weiterentwickeln? Und da kommt auf einmal ein neues Memoryhouse Album um die Ecke. Ich habe sie mehr …</p><p><img alt="Soft Hate" src="http://xsteadfastx.org/images/memoryhouse-softhate.jpg"></p> <p>Eigentlich dachte ich das es das war mit Dreampop/Chillwave. 5 bis 6 Jahre und man denkt man hat alles gehört. Jedes 90er Zitat bis zum erbrechen gebracht. Wie kann man sich da weiterentwickeln? Und da kommt auf einmal ein neues Memoryhouse Album um die Ecke. Ich habe sie mehr vermisst als ich erst dachte.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/r8erUBLmAnY' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Machs gut Marietta2016-02-08T13:11:00+01:002016-02-08T13:11:00+01:00marvintag:xsteadfastx.org,2016-02-08:/2016/02/08/machs-gut-marietta/<p>Und dann ist da der Moment in dem man merkt das sich die Band aufgelöst hat die man doch so gerne mal live gesehen hätte. Mariettas beide Alben sind die beiden Platten die ich das letzte halbe Jahr vielleicht am meisten gehört habe. Ein Grund zu hoffen das Emo doch …</p><p>Und dann ist da der Moment in dem man merkt das sich die Band aufgelöst hat die man doch so gerne mal live gesehen hätte. Mariettas beide Alben sind die beiden Platten die ich das letzte halbe Jahr vielleicht am meisten gehört habe. Ein Grund zu hoffen das Emo doch überlebt hat. Und dann sind sie auf einmal wieder weg. Was bleibt? Ein tolles Live-Video von ihrer letzten Show. Läuft schon das ganze Wochenende durch.</p> <p><span class="videobox"> <iframe width="640" height="390" src='https://www.youtube.com/embed/MD-6RKxCdTM' frameborder='0' webkitAllowFullScreen mozallowfullscreen allowFullScreen> </iframe> </span></p>Fernsehserien, Trakt, Jupyter und Bokeh2016-02-05T15:48:00+01:002016-02-05T15:48:00+01:00marvintag:xsteadfastx.org,2016-02-05:/2016/02/05/fernsehserien-trakt-jupyter-und-bokeh/<p> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Datenvisualisierung hat mich schon immer fasziniert. Leider hatte ich nie einen wirklichen Grund dafür es einzusetzen. Und doch wollte ich es mal ausprobieren. Ich habe mich sehr von <a href="http://blog.jemu.name/tvshows/2015/04/so-cougar-town-ended/">Jemus</a> inspirieren lassen. Da passiert natürlich viel viel mehr und ausführlicher. Ich habe erstmal das gemacht was ich auch verstehe :). Und zwar …</p></div></div></div></p><p> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Datenvisualisierung hat mich schon immer fasziniert. Leider hatte ich nie einen wirklichen Grund dafür es einzusetzen. Und doch wollte ich es mal ausprobieren. Ich habe mich sehr von <a href="http://blog.jemu.name/tvshows/2015/04/so-cougar-town-ended/">Jemus</a> inspirieren lassen. Da passiert natürlich viel viel mehr und ausführlicher. Ich habe erstmal das gemacht was ich auch verstehe :). Und zwar die Episoden Ratings darzustellen.</p> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Als erstes importiere ich ein paar Sachen. Erstmal <code>choice</code> um per Zufall Farben aus einer Palette auszuwählen. Dann <code>pandas</code> für die <code>DataFrames</code>. Um die Daten aus der Trakt-API zu bekommen benutze ich <a href="https://github.com/fuzeman/trakt.py">trakt.py</a>. Wieso alles selber bauen wenn es dieses schöne Modul gibt? Dann benutze ich <a href="http://bokeh.pydata.org/en/latest/">Bokeh</a> um das ganze darzustellen.</p> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[1]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="kn">from</span> <span class="nn">random</span> <span class="kn">import</span> <span class="n">choice</span> <span class="kn">import</span> <span class="nn">pandas</span> <span class="kn">as</span> <span class="nn">pd</span> <span class="kn">from</span> <span class="nn">trakt.tv</span> <span class="kn">import</span> <span class="n">TVShow</span> <span class="kn">from</span> <span class="nn">bokeh.plotting</span> <span class="kn">import</span> <span class="n">figure</span><span class="p">,</span> <span class="n">ColumnDataSource</span><span class="p">,</span> <span class="n">output_notebook</span><span class="p">,</span> <span class="n">show</span> <span class="kn">from</span> <span class="nn">bokeh.models</span> <span class="kn">import</span> <span class="n">HoverTool</span> <span class="kn">from</span> <span class="nn">bokeh.palettes</span> <span class="kn">import</span> <span class="n">Spectral9</span> </pre></div> </div> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Wir starten die Bokeh Ausgabe in einem jupyter-Notebook.</p> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[2]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="n">output_notebook</span><span class="p">()</span> </pre></div> </div> </div> </div> <div class="output_wrapper"> <div class="output"> <div class="output_area"><div class="prompt"></div> <div class="output_html rendered_html output_subarea "> <script type="text/javascript"> (function(global) { function now() { return new Date(); } if (typeof (window._bokeh_onload_callbacks) === "undefined") { window._bokeh_onload_callbacks = []; } function run_callbacks() { window._bokeh_onload_callbacks.forEach(function(callback) { callback() }); delete window._bokeh_onload_callbacks console.info("Bokeh: all callbacks have finished"); } function load_libs(js_urls, callback) { window._bokeh_onload_callbacks.push(callback); if (window._bokeh_is_loading > 0) { console.log("Bokeh: BokehJS is being loaded, scheduling callback at", now()); return null; } if (js_urls == null || js_urls.length === 0) { run_callbacks(); return null; } console.log("Bokeh: BokehJS not loaded, scheduling load and callback at", now()); window._bokeh_is_loading = js_urls.length; for (var i = 0; i < js_urls.length; i++) { var url = js_urls[i]; var s = document.createElement('script'); s.src = url; s.async = false; s.onreadystatechange = s.onload = function() { window._bokeh_is_loading--; if (window._bokeh_is_loading === 0) { console.log("Bokeh: all BokehJS libraries loaded"); run_callbacks() } }; s.onerror = function() { console.warn("failed to load library " + url); }; console.log("Bokeh: injecting script tag for BokehJS library: ", url); document.getElementsByTagName("head")[0].appendChild(s); } };var js_urls = ['https://cdn.pydata.org/bokeh/release/bokeh-0.11.0.min.js', 'https://cdn.pydata.org/bokeh/release/bokeh-widgets-0.11.0.min.js', 'https://cdn.pydata.org/bokeh/release/bokeh-compiler-0.11.0.min.js']; var inline_js = [ function(Bokeh) { Bokeh.set_log_level("info"); }, function(Bokeh) { console.log("Bokeh: injecting CSS: https://cdn.pydata.org/bokeh/release/bokeh-0.11.0.min.css"); Bokeh.embed.inject_css("https://cdn.pydata.org/bokeh/release/bokeh-0.11.0.min.css"); console.log("Bokeh: injecting CSS: https://cdn.pydata.org/bokeh/release/bokeh-widgets-0.11.0.min.css"); Bokeh.embed.inject_css("https://cdn.pydata.org/bokeh/release/bokeh-widgets-0.11.0.min.css"); } ]; function run_inline_js() { for (var i = 0; i < inline_js.length; i++) { inline_js[i](window.Bokeh); } } if (window._bokeh_is_loading === 0) { console.log("Bokeh: BokehJS loaded, going straight to plotting"); run_inline_js(); } else { load_libs(js_urls, function() { console.log("Bokeh: BokehJS plotting callback run at", now()); run_inline_js(); }); } }(this)); </script> <div> <a href="http://bokeh.pydata.org" target="_blank" class="bk-logo bk-logo-small bk-logo-notebook"></a> <span>BokehJS successfully loaded.</span> </div> </div> </div> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Hier ist die Funktion die die Daten besorgt, verarbeitet und darstellt.</p> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[3]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="k">def</span> <span class="nf">ratings</span><span class="p">(</span><span class="n">show_name</span><span class="p">):</span> <span class="c1"># Es wird die Show definiert</span> <span class="n">tv_show</span> <span class="o">=</span> <span class="n">TVShow</span><span class="p">(</span><span class="n">show_name</span><span class="p">)</span> <span class="c1"># Um die Staffeln ausseinander zu halten, soll jede Episode eine unterschiedliche </span> <span class="c1"># Staffelfarbe haben. Bokeh hat ein paar Farbpaletten dabei. Ich benutze Spectral9.</span> <span class="c1"># Dazu brauchen wir noch eine liste um alle Episoden-Staffel-Farben zu speichern</span> <span class="n">colormap</span> <span class="o">=</span> <span class="n">Spectral9</span> <span class="n">colors</span> <span class="o">=</span> <span class="p">[]</span> <span class="c1"># Wir brauchen eine Episoden Liste</span> <span class="n">episodes</span> <span class="o">=</span> <span class="p">[]</span> <span class="c1"># Damit die Episoden hintereinander auf der X-Achse dargestellt werden, brauchen sie</span> <span class="c1"># eine fortlaufende Nummer</span> <span class="n">episode_number</span> <span class="o">=</span> <span class="mi">1</span> <span class="c1"># Dann arbeiten wir uns mal durch die Staffeln</span> <span class="k">for</span> <span class="n">season</span> <span class="ow">in</span> <span class="n">tv_show</span><span class="o">.</span><span class="n">seasons</span><span class="p">:</span> <span class="c1"># Season 0 beinhaltet die Specials. Die werden ingnoriert</span> <span class="k">if</span> <span class="n">season</span><span class="o">.</span><span class="n">season</span> <span class="o">==</span> <span class="mi">0</span><span class="p">:</span> <span class="k">continue</span> <span class="c1"># Eine Staffel-Farbe wird per Zufall ausgewaehlt. Wir schmeissen immer die letzte Farbe</span> <span class="c1"># aus einer Kopie der Farbenliste damit sie nicht nochmal gewaehlt wird</span> <span class="k">if</span> <span class="n">colors</span><span class="p">:</span> <span class="n">new_colormap</span> <span class="o">=</span> <span class="n">colormap</span><span class="p">[:]</span> <span class="n">new_colormap</span><span class="o">.</span><span class="n">remove</span><span class="p">(</span><span class="n">colors</span><span class="p">[</span><span class="o">-</span><span class="mi">1</span><span class="p">])</span> <span class="n">color</span> <span class="o">=</span> <span class="n">choice</span><span class="p">(</span><span class="n">new_colormap</span><span class="p">)</span> <span class="k">else</span><span class="p">:</span> <span class="n">color</span> <span class="o">=</span> <span class="n">choice</span><span class="p">(</span><span class="n">colormap</span><span class="p">)</span> <span class="c1"># Die Episoden </span> <span class="k">for</span> <span class="n">episode</span> <span class="ow">in</span> <span class="n">season</span><span class="o">.</span><span class="n">episodes</span><span class="p">:</span> <span class="c1"># Wenn es eine Ausstrahlungsdatum gibt, wird der Zeit-Teil weggeschnitten</span> <span class="k">if</span> <span class="n">episode</span><span class="o">.</span><span class="n">first_aired</span><span class="p">:</span> <span class="n">aired</span> <span class="o">=</span> <span class="n">episode</span><span class="o">.</span><span class="n">first_aired</span><span class="o">.</span><span class="n">split</span><span class="p">(</span><span class="s1">&#39;T&#39;</span><span class="p">)[</span><span class="mi">0</span><span class="p">]</span> <span class="k">else</span><span class="p">:</span> <span class="n">aired</span> <span class="o">=</span> <span class="bp">None</span> <span class="c1"># Nun werden die gesammelten Daten als Tupel der Episodenliste angehangen</span> <span class="n">episodes</span><span class="o">.</span><span class="n">append</span><span class="p">((</span><span class="n">episode_number</span><span class="p">,</span> <span class="n">episode</span><span class="o">.</span><span class="n">season</span><span class="p">,</span> <span class="s1">&#39;{}.{}&#39;</span><span class="o">.</span><span class="n">format</span><span class="p">(</span><span class="n">episode</span><span class="o">.</span><span class="n">season</span><span class="p">,</span> <span class="n">episode</span><span class="o">.</span><span class="n">number</span><span class="p">),</span> <span class="n">episode</span><span class="o">.</span><span class="n">title</span><span class="p">,</span> <span class="n">aired</span><span class="p">,</span> <span class="n">episode</span><span class="o">.</span><span class="n">rating</span><span class="p">))</span> <span class="c1"># Die Episoden-Staffel-Farbe wird der Farbenliste angehangen</span> <span class="n">colors</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="n">color</span><span class="p">)</span> <span class="c1"># Die Episodennummer wird erhoeht</span> <span class="n">episode_number</span> <span class="o">+=</span> <span class="mi">1</span> <span class="c1"># Es wird ein DataFrame erstellt</span> <span class="n">df</span> <span class="o">=</span> <span class="n">pd</span><span class="o">.</span><span class="n">DataFrame</span><span class="p">(</span> <span class="n">episodes</span><span class="p">,</span> <span class="n">columns</span><span class="o">=</span><span class="p">[</span><span class="s1">&#39;episode_number&#39;</span><span class="p">,</span> <span class="s1">&#39;season&#39;</span><span class="p">,</span> <span class="s1">&#39;episode&#39;</span><span class="p">,</span> <span class="s1">&#39;title&#39;</span><span class="p">,</span> <span class="s1">&#39;aired&#39;</span><span class="p">,</span> <span class="s1">&#39;rating&#39;</span><span class="p">])</span> <span class="c1"># Es wird ein source-Objekt gebraucht damit die tooltips funktionieren.</span> <span class="c1"># Der DataFrame genuegt nicht um die Daten fuer die tooltips zu benutzen</span> <span class="n">source</span> <span class="o">=</span> <span class="n">ColumnDataSource</span><span class="p">(</span><span class="n">data</span><span class="o">=</span><span class="n">df</span><span class="p">)</span> <span class="c1"># Wir erstellen in figure und belabeln die Achsen</span> <span class="n">fig</span> <span class="o">=</span> <span class="n">figure</span><span class="p">(</span><span class="n">title</span><span class="o">=</span><span class="n">show_name</span><span class="p">)</span> <span class="n">fig</span><span class="o">.</span><span class="n">xaxis</span><span class="o">.</span><span class="n">axis_label</span> <span class="o">=</span> <span class="s1">&#39;Episodes&#39;</span> <span class="n">fig</span><span class="o">.</span><span class="n">yaxis</span><span class="o">.</span><span class="n">axis_label</span> <span class="o">=</span> <span class="s1">&#39;Rating&#39;</span> <span class="c1"># Hier bauen wir den Scatter-Plot. Wir uebergeben X und Y plus unser source-Objekt</span> <span class="c1"># und Style Geschichten. Unter anderem unsere Episoden-Farbenliste</span> <span class="n">fig</span><span class="o">.</span><span class="n">scatter</span><span class="p">(</span><span class="n">df</span><span class="p">[</span><span class="s1">&#39;episode_number&#39;</span><span class="p">],</span> <span class="n">df</span><span class="p">[</span><span class="s1">&#39;rating&#39;</span><span class="p">],</span> <span class="n">source</span><span class="o">=</span><span class="n">source</span><span class="p">,</span> <span class="n">name</span><span class="o">=</span><span class="s1">&#39;main&#39;</span><span class="p">,</span> <span class="n">fill_alpha</span><span class="o">=</span><span class="mf">0.8</span><span class="p">,</span> <span class="n">line_color</span><span class="o">=</span><span class="s1">&#39;#000000&#39;</span><span class="p">,</span> <span class="n">size</span><span class="o">=</span><span class="mi">10</span><span class="p">,</span> <span class="n">color</span><span class="o">=</span><span class="n">colors</span><span class="p">)</span> <span class="c1"># Ein HoverTool-Object in dem wir das Layout fuer die Tooltips festlegen</span> <span class="n">hover</span> <span class="o">=</span> <span class="n">HoverTool</span><span class="p">(</span><span class="n">tooltips</span><span class="o">=</span><span class="p">[(</span><span class="s1">&#39;Episode&#39;</span><span class="p">,</span> <span class="s1">&#39;@episode&#39;</span><span class="p">),</span> <span class="p">(</span><span class="s1">&#39;Title&#39;</span><span class="p">,</span> <span class="s1">&#39;@title&#39;</span><span class="p">),</span> <span class="p">(</span><span class="s1">&#39;Rating&#39;</span><span class="p">,</span> <span class="s1">&#39;@rating&#39;</span><span class="p">),</span> <span class="p">(</span><span class="s1">&#39;Aired&#39;</span><span class="p">,</span> <span class="s1">&#39;@aired&#39;</span><span class="p">)])</span> <span class="c1"># Das Tool wird hinzugefuegt</span> <span class="n">fig</span><span class="o">.</span><span class="n">add_tools</span><span class="p">(</span><span class="n">hover</span><span class="p">)</span> <span class="c1"># Und nun angezeigt</span> <span class="n">show</span><span class="p">(</span><span class="n">fig</span><span class="p">)</span> </pre></div> </div> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Ich gebe eigentlich nicht viel auf die Ratings. Ich selber habe über Trakt noch nie etwas bewertet. Trotzdem handelt es sich um gute Daten um ein paar Sachen auszuprobieren.</p> <p>Hier ein paar Ratings von Serien die mit entweder viel bedeuten oder die ich gerade schaue.</p> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <h2 id="Roswell">Roswell<a class="anchor-link" href="#Roswell">&#182;</a></h2> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[4]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="n">ratings</span><span class="p">(</span><span class="s1">&#39;Roswell&#39;</span><span class="p">)</span> </pre></div> </div> </div> </div> <div class="output_wrapper"> <div class="output"> <div class="output_area"><div class="prompt"></div> <div class="output_html rendered_html output_subarea "> <div class="plotdiv" id="c003301d-6be0-4af4-9d83-a645595d71f8"></div> <script type="text/javascript"> (function(global) { function now() { return new Date(); } if (typeof (window._bokeh_onload_callbacks) === "undefined") { window._bokeh_onload_callbacks = []; } function run_callbacks() { window._bokeh_onload_callbacks.forEach(function(callback) { callback() }); delete window._bokeh_onload_callbacks console.info("Bokeh: all callbacks have finished"); } function load_libs(js_urls, callback) { window._bokeh_onload_callbacks.push(callback); if (window._bokeh_is_loading > 0) { console.log("Bokeh: BokehJS is being loaded, scheduling callback at", now()); return null; } if (js_urls == null || js_urls.length === 0) { run_callbacks(); return null; } console.log("Bokeh: BokehJS not loaded, scheduling load and callback at", now()); window._bokeh_is_loading = js_urls.length; for (var i = 0; i < js_urls.length; i++) { var url = js_urls[i]; var s = document.createElement('script'); s.src = url; s.async = false; s.onreadystatechange = s.onload = function() { window._bokeh_is_loading--; if (window._bokeh_is_loading === 0) { console.log("Bokeh: all BokehJS libraries loaded"); run_callbacks() } }; s.onerror = function() { console.warn("failed to load library " + url); }; console.log("Bokeh: injecting script tag for BokehJS library: ", url); document.getElementsByTagName("head")[0].appendChild(s); } };var element = document.getElementById("c003301d-6be0-4af4-9d83-a645595d71f8"); if (element == null) { console.log("Bokeh: ERROR: autoload.js configured with elementid 'c003301d-6be0-4af4-9d83-a645595d71f8' but no matching script tag was found. ") return false; }var js_urls = []; var inline_js = [ function(Bokeh) { Bokeh.$(function() { var docs_json = {"a4444eeb-7dcd-4c07-b94e-3ad610dd856b": {"title": "Bokeh Application", "version": "0.11.0", "roots": {"references": [{"type": "BasicTicker", "attributes": {}, "id": "fe11e353-2325-48f3-b384-a02bf4b62892"}, {"type": "ResizeTool", "attributes": {"plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}}, "id": "9705b83f-1b1a-43ee-b1d1-e205d468c0fd"}, {"type": "Circle", "attributes": {"x": {"field": "x"}, "size": {"value": 10, "units": "screen"}, "line_alpha": {"value": 0.1}, "fill_alpha": {"value": 0.1}, "fill_color": {"value": "#1f77b4"}, "y": {"field": "y"}, "line_color": {"value": "#1f77b4"}}, "id": "d2a09c3d-4ac7-415c-93ac-b422f933cc07"}, {"type": "Grid", "attributes": {"ticker": {"type": "BasicTicker", "id": "fe11e353-2325-48f3-b384-a02bf4b62892"}, "plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}}, "id": "e907c445-c849-455f-8196-6733edaf9abb"}, {"type": "LinearAxis", "attributes": {"axis_label": "Episodes", "ticker": {"type": "BasicTicker", "id": "fe11e353-2325-48f3-b384-a02bf4b62892"}, "plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}, "formatter": {"type": "BasicTickFormatter", "id": "83d98f0e-de6c-44e2-8474-19e044bbb2e3"}}, "id": "37234061-b115-4cc7-8743-19dfb6adf097"}, {"type": "DataRange1d", "attributes": {"callback": null}, "id": "a19547b0-d274-41f1-9a9e-f8b417a6f60d"}, {"type": "HoverTool", "attributes": {"tooltips": [["Episode", "@episode"], ["Title", "@title"], ["Rating", "@rating"], ["Aired", "@aired"]], "plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}, "callback": null}, "id": "0f0313c1-ed74-4894-8329-42e4cc16c6a3"}, {"type": "BoxAnnotation", "attributes": {"right_units": "screen", "line_width": {"value": 2}, "top_units": "screen", "line_dash": [4, 4], "left_units": "screen", "line_color": {"value": "black"}, "plot": null, "line_alpha": {"value": 1.0}, "fill_alpha": {"value": 0.5}, "fill_color": {"value": "lightgrey"}, "bottom_units": "screen", "level": "overlay", "render_mode": "css"}, "id": "50e4b201-a32e-49f1-bc9e-aed3b05dbfb6"}, {"type": "BoxZoomTool", "attributes": {"plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}, "overlay": {"type": "BoxAnnotation", "id": "50e4b201-a32e-49f1-bc9e-aed3b05dbfb6"}}, "id": "a5c725ee-66d9-4ec6-9779-9e8c2455ac05"}, {"type": "GlyphRenderer", "attributes": {"glyph": {"type": "Circle", "id": "99a2bf0a-3638-4dc4-970a-7b9e5d1942d1"}, "hover_glyph": null, "selection_glyph": null, "nonselection_glyph": {"type": "Circle", "id": "d2a09c3d-4ac7-415c-93ac-b422f933cc07"}, "data_source": {"type": "ColumnDataSource", "id": "38f35675-cbb9-47d1-9540-a662f5ad1331"}, "name": "main"}, "id": "7039bbcf-edec-43e2-bec3-bdcf9c2c95a2"}, {"type": "Plot", "attributes": {"title": "Roswell", "below": [{"type": "LinearAxis", "id": "37234061-b115-4cc7-8743-19dfb6adf097"}], "y_range": {"type": "DataRange1d", "id": "653fed87-ae50-47f4-adf9-6b9e998feb22"}, "x_range": {"type": "DataRange1d", "id": "a19547b0-d274-41f1-9a9e-f8b417a6f60d"}, "tools": [{"type": "PanTool", "id": "13321860-4eb1-4b4d-8b7d-ba375375106a"}, {"type": "WheelZoomTool", "id": "d3986be7-ff9d-454e-a87d-a7b8bdce9f1a"}, {"type": "BoxZoomTool", "id": "a5c725ee-66d9-4ec6-9779-9e8c2455ac05"}, {"type": "PreviewSaveTool", "id": "a3755ffb-f36b-4a1c-b51a-8b18de101320"}, {"type": "ResizeTool", "id": "9705b83f-1b1a-43ee-b1d1-e205d468c0fd"}, {"type": "ResetTool", "id": "3f5c5537-a4fa-45df-a453-c4de327c4465"}, {"type": "HelpTool", "id": "1c55ef60-50c8-4950-baaf-0c5c5d050785"}, {"type": "HoverTool", "id": "0f0313c1-ed74-4894-8329-42e4cc16c6a3"}], "left": [{"type": "LinearAxis", "id": "e97fcc6c-13c1-48a4-8bdd-c87dddd39d53"}], "renderers": [{"type": "LinearAxis", "id": "37234061-b115-4cc7-8743-19dfb6adf097"}, {"type": "Grid", "id": "e907c445-c849-455f-8196-6733edaf9abb"}, {"type": "LinearAxis", "id": "e97fcc6c-13c1-48a4-8bdd-c87dddd39d53"}, {"type": "Grid", "id": "db491334-712f-4782-92d6-552d53352382"}, {"type": "BoxAnnotation", "id": "50e4b201-a32e-49f1-bc9e-aed3b05dbfb6"}, {"type": "GlyphRenderer", "id": "7039bbcf-edec-43e2-bec3-bdcf9c2c95a2"}], "tool_events": {"type": "ToolEvents", "id": "e5c02ecb-352a-4e60-a041-19b5094799aa"}}, "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}, {"type": "BasicTickFormatter", "attributes": {}, "id": "83d98f0e-de6c-44e2-8474-19e044bbb2e3"}, {"type": "HelpTool", "attributes": {"plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}}, "id": "1c55ef60-50c8-4950-baaf-0c5c5d050785"}, {"type": "PanTool", "attributes": {"plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}}, "id": "13321860-4eb1-4b4d-8b7d-ba375375106a"}, {"type": "ToolEvents", "attributes": {}, "id": "e5c02ecb-352a-4e60-a041-19b5094799aa"}, {"type": "DataRange1d", "attributes": {"callback": null}, "id": "653fed87-ae50-47f4-adf9-6b9e998feb22"}, {"type": "LinearAxis", "attributes": {"axis_label": "Rating", "ticker": {"type": "BasicTicker", "id": "aefcc403-2711-4ab2-8db5-f885c1df8ca2"}, "plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}, "formatter": {"type": "BasicTickFormatter", "id": "c3d4b516-9ab6-4b46-a0ef-b9b69b770f2f"}}, "id": "e97fcc6c-13c1-48a4-8bdd-c87dddd39d53"}, {"type": "WheelZoomTool", "attributes": {"plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}}, "id": "d3986be7-ff9d-454e-a87d-a7b8bdce9f1a"}, {"type": "Grid", "attributes": {"dimension": 1, "ticker": {"type": "BasicTicker", "id": "aefcc403-2711-4ab2-8db5-f885c1df8ca2"}, "plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}}, "id": "db491334-712f-4782-92d6-552d53352382"}, {"type": "BasicTicker", "attributes": {}, "id": "aefcc403-2711-4ab2-8db5-f885c1df8ca2"}, {"type": "ColumnDataSource", "attributes": {"column_names": ["title", "episode_number", "rating", "episode", "index", "aired", "season", "x", "y", "fill_color"], "data": {"title": ["Pilot", "The Morning After", "Monsters", "Leaving Normal", "Missing", "285 South (1)", "River Dog (2)", "Blood Brother", "Heat Wave", "The Balance", "Toy House", "Into the Woods", "The Convention", "Blind Date", "Independence Day", "Sexual Healing", "Crazy", "Tess, Lies and Videotape", "Four Square", "Max to the Max", "The White Room (1)", "Destiny (2)", "Skin and Bones", "Ask Not", "Surprise", "Summer of '47", "The End of the World", "Harvest", "Wipeout!", "Meet the Dupes (1)", "Max in the City (2)", "A Roswell Christmas Carol", "To Serve and Protect", "We Are Family", "Disturbing Behavior (1)", "How the Other Half Lives (2)", "Viva Las Vegas", "Heart of Mine", "Cry Your Name", "It's Too Late and It's Too Bad", "Baby, It's You", "Off the Menu", "The Departure", "Busted", "Michael, The Guys and The Great Snapple Caper", "Significant Others", "Secrets and Lies (1)", "Control (2)", "To Have and To Hold", "Interruptus", "Behind the Music", "Samuel Rising", "A Tale of Two Parties", "I Married An Alien", "Ch-Ch-Changes", "Panacea", "Chant Down Babylon", "Who Died and Made You King?", "Crash", "Four Aliens and A Baby", "Graduation"], "x": [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61], "y": [8.15625, 7.74468, 8.0, 8.20588, 7.80645, 8.38461, 8.28, 8.30769, 8.46154, 8.0, 7.95652, 8.13043, 7.90909, 8.26923, 8.13636, 8.45833, 8.13636, 8.42857, 8.33333, 8.5, 8.33333, 8.69565, 7.89474, 8.41177, 8.375, 7.89474, 8.3913, 8.41177, 7.8125, 7.9, 7.92857, 8.0, 7.82353, 8.05882, 7.9375, 7.9375, 8.27778, 8.05556, 8.11765, 8.1875, 8.35714, 8.07692, 8.46154, 7.5, 7.84615, 8.09091, 8.11111, 8.33333, 8.2, 8.16667, 7.88889, 7.66667, 8.16667, 8.54545, 8.1, 8.45455, 7.11111, 7.9, 8.375, 8.5, 8.92308], "season": [1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3], "rating": [8.15625, 7.74468, 8.0, 8.20588, 7.80645, 8.38461, 8.28, 8.30769, 8.46154, 8.0, 7.95652, 8.13043, 7.90909, 8.26923, 8.13636, 8.45833, 8.13636, 8.42857, 8.33333, 8.5, 8.33333, 8.69565, 7.89474, 8.41177, 8.375, 7.89474, 8.3913, 8.41177, 7.8125, 7.9, 7.92857, 8.0, 7.82353, 8.05882, 7.9375, 7.9375, 8.27778, 8.05556, 8.11765, 8.1875, 8.35714, 8.07692, 8.46154, 7.5, 7.84615, 8.09091, 8.11111, 8.33333, 8.2, 8.16667, 7.88889, 7.66667, 8.16667, 8.54545, 8.1, 8.45455, 7.11111, 7.9, 8.375, 8.5, 8.92308], "fill_color": ["#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61"], "episode": ["1.1", "1.2", "1.3", "1.4", "1.5", "1.6", "1.7", "1.8", "1.9", "1.10", "1.11", "1.12", "1.13", "1.14", "1.15", "1.16", "1.17", "1.18", "1.19", "1.20", "1.21", "1.22", "2.1", "2.2", "2.3", "2.4", "2.5", "2.6", "2.7", "2.8", "2.9", "2.10", "2.11", "2.12", "2.13", "2.14", "2.15", "2.16", "2.17", "2.18", "2.19", "2.20", "2.21", "3.1", "3.2", "3.3", "3.4", "3.5", "3.6", "3.7", "3.8", "3.9", "3.10", "3.11", "3.12", "3.13", "3.14", "3.15", "3.16", "3.17", "3.18"], "index": [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60], "aired": ["1999-10-07", "1999-10-14", "1999-10-21", "1999-10-28", "1999-11-04", "1999-11-11", "1999-11-18", "1999-11-25", "1999-12-02", "1999-12-16", "2000-01-20", "2000-01-27", "2000-02-03", "2000-02-10", "2000-02-17", "2000-03-02", "2000-04-11", "2000-04-18", "2000-04-25", "2000-05-02", "2000-05-09", "2000-05-16", "2000-10-03", "2000-10-10", "2000-10-17", "2000-10-24", "2000-10-31", "2000-11-07", "2000-11-14", "2000-11-21", "2000-11-28", "2000-12-19", "2001-01-23", "2001-01-30", "2001-02-06", "2001-02-20", "2001-02-27", "2001-04-17", "2001-04-24", "2001-05-01", "2001-05-08", "2001-05-15", "2001-05-22", "2001-10-10", "2001-10-17", "2001-10-24", "2001-10-31", "2001-11-07", "2001-11-14", "2001-11-21", "2001-11-28", "2001-12-19", "2002-01-02", "2002-01-30", "2002-02-06", "2002-02-13", "2002-02-27", "2002-04-24", "2002-05-01", "2002-05-08", "2002-05-15"], "episode_number": [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61]}, "callback": null}, "id": "38f35675-cbb9-47d1-9540-a662f5ad1331"}, {"type": "BasicTickFormatter", "attributes": {}, "id": "c3d4b516-9ab6-4b46-a0ef-b9b69b770f2f"}, {"type": "ResetTool", "attributes": {"plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}}, "id": "3f5c5537-a4fa-45df-a453-c4de327c4465"}, {"type": "PreviewSaveTool", "attributes": {"plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}}, "id": "a3755ffb-f36b-4a1c-b51a-8b18de101320"}, {"type": "Circle", "attributes": {"x": {"field": "x"}, "size": {"value": 10, "units": "screen"}, "fill_alpha": {"value": 0.8}, "fill_color": {"field": "fill_color"}, "y": {"field": "y"}, "line_color": {"value": "#000000"}}, "id": "99a2bf0a-3638-4dc4-970a-7b9e5d1942d1"}], "root_ids": ["63fdc565-8be5-4ae8-a4c1-7c0549348629"]}}}; var render_items = [{"docid": "a4444eeb-7dcd-4c07-b94e-3ad610dd856b", "modelid": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "notebook_comms_target": "395941fd-3b4a-4bdc-a1e0-2b654571cfc7", "elementid": "c003301d-6be0-4af4-9d83-a645595d71f8"}]; Bokeh.embed.embed_items(docs_json, render_items); }); }, function(Bokeh) { } ]; function run_inline_js() { for (var i = 0; i < inline_js.length; i++) { inline_js[i](window.Bokeh); } } if (window._bokeh_is_loading === 0) { console.log("Bokeh: BokehJS loaded, going straight to plotting"); run_inline_js(); } else { load_libs(js_urls, function() { console.log("Bokeh: BokehJS plotting callback run at", now()); run_inline_js(); }); } }(this)); </script> </div> </div> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <p>Die erste Serie bei der ich von einem persönlichen Fandom sprechen kann. Noch bevor ich wusste das es das Wort gibt. Ich habe damals sogar alle Bücher gelesen. Kurz hatte ich nachgedacht sie wegzuschmeissen. Leicht peinlich berührt meiner Serienliebe von damals. Habe mich doch dagegen entschieden. Die erste Staffel war grossartig. Der perfekte Mix aus Highschool und SciFi/Mystery. Danach nahm es für mich drastisch ab. Die dritte Staffel habe ich noch nicht einmal schauen können. Zu gross war die Angst vor dem Zerstören meiner Liebe zur Serie.</p> </div> </div> </div> <div class="cell border-box-sizing text_cell rendered"> <div class="prompt input_prompt"> </div> <div class="inner_cell"> <div class="text_cell_render border-box-sizing rendered_html"> <h2 id="Veronica-Mars">Veronica Mars<a class="anchor-link" href="#Veronica-Mars">&#182;</a></h2> </div> </div> </div> <div class="cell border-box-sizing code_cell rendered"> <div class="input"> <div class="prompt input_prompt">In&nbsp;[5]:</div> <div class="inner_cell"> <div class="input_area"> <div class="highlight-ipynb"><pre class="ipynb"><span></span><span class="n">ratings</span><span class="p">(</span><span class="s1">&#39;Veronica Mars&#39;</span><span class="p">)</span> </pre></div> </div> </div> </div> <div class="output_wrapper"> <div class="output"> <div class="output_area"><div class="prompt"></div> <div class="output_html rendered_html output_subarea "> <div class="plotdiv" id="0ff0b9c2-c47f-4c8e-8fff-7c8263552fc8"></div> <script type="text/javascript"> (function(global) { function now() { return new Date(); } if (typeof (window._bokeh_onload_callbacks) === "undefined") { window._bokeh_onload_callbacks = []; } function run_callbacks() { window._bokeh_onload_callbacks.forEach(function(callback) { callback() }); delete window._bokeh_onload_callbacks console.info("Bokeh: all callbacks have finished"); } function load_libs(js_urls, callback) { window._bokeh_onload_callbacks.push(callback); if (window._bokeh_is_loading > 0) { console.log("Bokeh: BokehJS is being loaded, scheduling callback at", now()); return null; } if (js_urls == null || js_urls.length === 0) { run_callbacks(); return null; } console.log("Bokeh: BokehJS not loaded, scheduling load and callback at", now()); window._bokeh_is_loading = js_urls.length; for (var i = 0; i < js_urls.length; i++) { var url = js_urls[i]; var s = document.createElement('script'); s.src = url; s.async = false; s.onreadystatechange = s.onload = function() { window._bokeh_is_loading--; if (window._bokeh_is_loading === 0) { console.log("Bokeh: all BokehJS libraries loaded"); run_callbacks() } }; s.onerror = function() { console.warn("failed to load library " + url); }; console.log("Bokeh: injecting script tag for BokehJS library: ", url); document.getElementsByTagName("head")[0].appendChild(s); } };var element = document.getElementById("0ff0b9c2-c47f-4c8e-8fff-7c8263552fc8"); if (element == null) { console.log("Bokeh: ERROR: autoload.js configured with elementid '0ff0b9c2-c47f-4c8e-8fff-7c8263552fc8' but no matching script tag was found. ") return false; }var js_urls = []; var inline_js = [ function(Bokeh) { Bokeh.$(function() { var docs_json = {"34d23b52-0e47-4598-af48-49ae35b9e885": {"title": "Bokeh Application", "version": "0.11.0", "roots": {"references": [{"type": "PreviewSaveTool", "attributes": {"plot": {"type": "Plot", "id": "483f0035-81b9-4a13-988b-7186df428d57", "subtype": "Figure"}}, "id": "9ac9f239-3742-4410-9210-ebbec89a235c"}, {"type": "BasicTicker", "attributes": {}, "id": "fe11e353-2325-48f3-b384-a02bf4b62892"}, {"type": "Circle", "attributes": {"x": {"field": "x"}, "size": {"value": 10, "units": "screen"}, "line_alpha": {"value": 0.1}, "fill_alpha": {"value": 0.1}, "fill_color": {"value": "#1f77b4"}, "y": {"field": "y"}, "line_color": {"value": "#1f77b4"}}, "id": "ea49217a-64eb-41e9-af46-ccc170c93f1c"}, {"type": "ResizeTool", "attributes": {"plot": {"type": "Plot", "id": "483f0035-81b9-4a13-988b-7186df428d57", "subtype": "Figure"}}, "id": "55909f10-136d-4d35-a07d-bc0b17dc21d0"}, {"type": "BasicTicker", "attributes": {}, "id": "dd2909ed-5221-450e-b0ba-06357ae79532"}, {"type": "PanTool", "attributes": {"plot": {"type": "Plot", "id": "483f0035-81b9-4a13-988b-7186df428d57", "subtype": "Figure"}}, "id": "15b54b64-7d0d-45bb-bbae-a58ce69436e9"}, {"type": "Circle", "attributes": {"x": {"field": "x"}, "size": {"value": 10, "units": "screen"}, "fill_alpha": {"value": 0.8}, "fill_color": {"field": "fill_color"}, "y": {"field": "y"}, "line_color": {"value": "#000000"}}, "id": "d4155969-83f5-404f-9994-240fb367b9fa"}, {"type": "DataRange1d", "attributes": {"callback": null}, "id": "ffaf50e2-7b15-4b4b-bdd7-4652fa735c9f"}, {"type": "LinearAxis", "attributes": {"axis_label": "Episodes", "ticker": {"type": "BasicTicker", "id": "fe11e353-2325-48f3-b384-a02bf4b62892"}, "plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}, "formatter": {"type": "BasicTickFormatter", "id": "83d98f0e-de6c-44e2-8474-19e044bbb2e3"}}, "id": "37234061-b115-4cc7-8743-19dfb6adf097"}, {"type": "Circle", "attributes": {"x": {"field": "x"}, "size": {"value": 10, "units": "screen"}, "line_alpha": {"value": 0.1}, "fill_alpha": {"value": 0.1}, "fill_color": {"value": "#1f77b4"}, "y": {"field": "y"}, "line_color": {"value": "#1f77b4"}}, "id": "d2a09c3d-4ac7-415c-93ac-b422f933cc07"}, {"type": "BoxAnnotation", "attributes": {"right_units": "screen", "line_width": {"value": 2}, "top_units": "screen", "line_dash": [4, 4], "left_units": "screen", "line_color": {"value": "black"}, "plot": null, "line_alpha": {"value": 1.0}, "fill_alpha": {"value": 0.5}, "fill_color": {"value": "lightgrey"}, "bottom_units": "screen", "level": "overlay", "render_mode": "css"}, "id": "50e4b201-a32e-49f1-bc9e-aed3b05dbfb6"}, {"type": "BoxZoomTool", "attributes": {"plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}, "overlay": {"type": "BoxAnnotation", "id": "50e4b201-a32e-49f1-bc9e-aed3b05dbfb6"}}, "id": "a5c725ee-66d9-4ec6-9779-9e8c2455ac05"}, {"type": "WheelZoomTool", "attributes": {"plot": {"type": "Plot", "id": "483f0035-81b9-4a13-988b-7186df428d57", "subtype": "Figure"}}, "id": "a391a093-c6c0-4906-9e9d-14fab633db43"}, {"type": "LinearAxis", "attributes": {"axis_label": "Rating", "ticker": {"type": "BasicTicker", "id": "327a5c66-996b-4ee7-bad1-e60bf53332c0"}, "plot": {"type": "Plot", "id": "483f0035-81b9-4a13-988b-7186df428d57", "subtype": "Figure"}, "formatter": {"type": "BasicTickFormatter", "id": "a29344da-4ac0-4a8d-8c47-5e10c3db336c"}}, "id": "dff35f44-a4f1-4a49-8eb2-a0a865a26f8c"}, {"type": "DataRange1d", "attributes": {"callback": null}, "id": "545af528-44aa-440c-9176-c0fbef112836"}, {"type": "GlyphRenderer", "attributes": {"glyph": {"type": "Circle", "id": "d4155969-83f5-404f-9994-240fb367b9fa"}, "hover_glyph": null, "selection_glyph": null, "nonselection_glyph": {"type": "Circle", "id": "ea49217a-64eb-41e9-af46-ccc170c93f1c"}, "data_source": {"type": "ColumnDataSource", "id": "94d17170-6d72-46ee-8b60-a61bdd05d5ba"}, "name": "main"}, "id": "74a56789-be42-4c77-9354-26a946b9dc3f"}, {"type": "Grid", "attributes": {"dimension": 1, "ticker": {"type": "BasicTicker", "id": "327a5c66-996b-4ee7-bad1-e60bf53332c0"}, "plot": {"type": "Plot", "id": "483f0035-81b9-4a13-988b-7186df428d57", "subtype": "Figure"}}, "id": "0c818b40-e3a8-42ba-aee3-4b4f4f0c9f5e"}, {"type": "HoverTool", "attributes": {"tooltips": [["Episode", "@episode"], ["Title", "@title"], ["Rating", "@rating"], ["Aired", "@aired"]], "plot": {"type": "Plot", "id": "483f0035-81b9-4a13-988b-7186df428d57", "subtype": "Figure"}, "callback": null}, "id": "9c12a8e4-0bb3-445c-b5d7-b971985e88f6"}, {"type": "BasicTickFormatter", "attributes": {}, "id": "83d98f0e-de6c-44e2-8474-19e044bbb2e3"}, {"type": "HelpTool", "attributes": {"plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}}, "id": "1c55ef60-50c8-4950-baaf-0c5c5d050785"}, {"type": "GlyphRenderer", "attributes": {"glyph": {"type": "Circle", "id": "99a2bf0a-3638-4dc4-970a-7b9e5d1942d1"}, "hover_glyph": null, "selection_glyph": null, "nonselection_glyph": {"type": "Circle", "id": "d2a09c3d-4ac7-415c-93ac-b422f933cc07"}, "data_source": {"type": "ColumnDataSource", "id": "38f35675-cbb9-47d1-9540-a662f5ad1331"}, "name": "main"}, "id": "7039bbcf-edec-43e2-bec3-bdcf9c2c95a2"}, {"type": "DataRange1d", "attributes": {"callback": null}, "id": "653fed87-ae50-47f4-adf9-6b9e998feb22"}, {"type": "WheelZoomTool", "attributes": {"plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}}, "id": "d3986be7-ff9d-454e-a87d-a7b8bdce9f1a"}, {"type": "Grid", "attributes": {"dimension": 1, "ticker": {"type": "BasicTicker", "id": "aefcc403-2711-4ab2-8db5-f885c1df8ca2"}, "plot": {"type": "Plot", "id": "63fdc565-8be5-4ae8-a4c1-7c0549348629", "subtype": "Figure"}}, "id": "db491334-712f-4782-92d6-552d53352382"}, {"type": "BasicTickFormatter", "attributes": {}, "id": "a29344da-4ac0-4a8d-8c47-5e10c3db336c"}, {"type": "ColumnDataSource", "attributes": {"column_names": ["title", "episode_number", "rating", "episode", "index", "aired", "season", "x", "y", "fill_color"], "data": {"title": ["Pilot", "The Morning After", "Monsters", "Leaving Normal", "Missing", "285 South (1)", "River Dog (2)", "Blood Brother", "Heat Wave", "The Balance", "Toy House", "Into the Woods", "The Convention", "Blind Date", "Independence Day", "Sexual Healing", "Crazy", "Tess, Lies and Videotape", "Four Square", "Max to the Max", "The White Room (1)", "Destiny (2)", "Skin and Bones", "Ask Not", "Surprise", "Summer of '47", "The End of the World", "Harvest", "Wipeout!", "Meet the Dupes (1)", "Max in the City (2)", "A Roswell Christmas Carol", "To Serve and Protect", "We Are Family", "Disturbing Behavior (1)", "How the Other Half Lives (2)", "Viva Las Vegas", "Heart of Mine", "Cry Your Name", "It's Too Late and It's Too Bad", "Baby, It's You", "Off the Menu", "The Departure", "Busted", "Michael, The Guys and The Great Snapple Caper", "Significant Others", "Secrets and Lies (1)", "Control (2)", "To Have and To Hold", "Interruptus", "Behind the Music", "Samuel Rising", "A Tale of Two Parties", "I Married An Alien", "Ch-Ch-Changes", "Panacea", "Chant Down Babylon", "Who Died and Made You King?", "Crash", "Four Aliens and A Baby", "Graduation"], "x": [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61], "y": [8.15625, 7.74468, 8.0, 8.20588, 7.80645, 8.38461, 8.28, 8.30769, 8.46154, 8.0, 7.95652, 8.13043, 7.90909, 8.26923, 8.13636, 8.45833, 8.13636, 8.42857, 8.33333, 8.5, 8.33333, 8.69565, 7.89474, 8.41177, 8.375, 7.89474, 8.3913, 8.41177, 7.8125, 7.9, 7.92857, 8.0, 7.82353, 8.05882, 7.9375, 7.9375, 8.27778, 8.05556, 8.11765, 8.1875, 8.35714, 8.07692, 8.46154, 7.5, 7.84615, 8.09091, 8.11111, 8.33333, 8.2, 8.16667, 7.88889, 7.66667, 8.16667, 8.54545, 8.1, 8.45455, 7.11111, 7.9, 8.375, 8.5, 8.92308], "season": [1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3], "rating": [8.15625, 7.74468, 8.0, 8.20588, 7.80645, 8.38461, 8.28, 8.30769, 8.46154, 8.0, 7.95652, 8.13043, 7.90909, 8.26923, 8.13636, 8.45833, 8.13636, 8.42857, 8.33333, 8.5, 8.33333, 8.69565, 7.89474, 8.41177, 8.375, 7.89474, 8.3913, 8.41177, 7.8125, 7.9, 7.92857, 8.0, 7.82353, 8.05882, 7.9375, 7.9375, 8.27778, 8.05556, 8.11765, 8.1875, 8.35714, 8.07692, 8.46154, 7.5, 7.84615, 8.09091, 8.11111, 8.33333, 8.2, 8.16667, 7.88889, 7.66667, 8.16667, 8.54545, 8.1, 8.45455, 7.11111, 7.9, 8.375, 8.5, 8.92308], "fill_color": ["#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#ffffbf", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#e6f598", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61", "#fdae61"], "episode": ["1.1", "1.2", "1.3", "1.4", "1.5", "1.6"