Laden Sie ein Bild in dieses Stapel-Snippet und bewegen Sie die Maus darüber. Eine schwarze Kurve, die dem Farbtonwinkel ab dem Cursorpunkt folgt , wird gezeichnet:
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script><style>canvas{border:1px solid black;}</style>Load an image: <input type='file' onchange='load(this)'><br><br>Max length <input id='length' type='text' value='300'><br><br><div id='coords'></div><br><canvas id='c' width='100' height='100'>Your browser doesn't support the HTML5 canvas tag.</canvas><script>function load(t){if(t.files&&t.files[0]){var e=new FileReader;e.onload=setupImage,e.readAsDataURL(t.files[0])}}function setupImage(t){function e(t){t.attr("width",img.width),t.attr("height",img.height);var e=t[0].getContext("2d");return e.drawImage(img,0,0),e}img=$("<img>").attr("src",t.target.result)[0],ctx=e($("#c")),ctxRead=e($("<canvas>"))}function findPos(t){var e=0,a=0;if(t.offsetParent){do e+=t.offsetLeft,a+=t.offsetTop;while(t=t.offsetParent);return{x:e,y:a}}return void 0}$("#c").mousemove(function(t){function e(t,e){var a=ctxRead.getImageData(t,e,1,1).data,i=a[0]/255,r=a[1]/255,o=a[2]/255;return Math.atan2(Math.sqrt(3)*(r-o),2*i-r-o)}if("undefined"!=typeof img){var a=findPos(this),i=t.pageX-a.x,r=t.pageY-a.y;$("#coords").html("x = "+i.toString()+", y = "+r.toString());var o=parseInt($("#length").val());if(isNaN(o))return void alert("Bad max length!");for(var n=[i],f=[r],h=0;n[h]>=0&&n[h]<this.width&&f[h]>=0&&f[h]<this.height&&o>h;)n.push(n[h]+Math.cos(e(n[h],f[h]))),f.push(f[h]-Math.sin(e(n[h],f[h]))),h++;ctx.clearRect(0,0,this.width,this.height),ctx.drawImage(img,0,0);for(var h=0;h<n.length;h++)ctx.fillRect(Math.floor(n[h]),Math.floor(f[h]),1,1)}});</script>
Ich habe dieses Snippet nur in Google Chrome getestet.
Befindet sich der Cursor beispielsweise über Rot, weist die Kurve eine Neigung von 0 ° auf. Befindet er sich jedoch über Gelb, weist sie eine Neigung von 60 ° auf. Die Kurve wird für die angegebene Länge fortgesetzt, wobei die Steigung kontinuierlich geändert wird, um sie an den Farbton anzupassen.
Laden Sie dieses Bild und wenn Sie den Mauszeiger darüber bewegen, sollte die Linie um den Mauszeiger eine vollständige Drehung gegen den Uhrzeigersinn ausführen:
Dies und das sind andere nette Bilder zum Ausprobieren. (Sie müssen sie speichern und dann mit dem Snippet laden. Sie können aufgrund herkunftsübergreifender Einschränkungen nicht direkt verknüpft werden.)
Hier ist eine nicht verkleinerte Version des Snippets:
Herausforderung
Schreiben Sie ein Programm, das das tut, was das Snippet tut, nur nicht interaktiv. Nehmen Sie ein Bild und eine (x, y) Koordinate in den Bildgrenzen sowie eine maximale Kurvenlänge auf. Das gleiche Bild mit der hinzugefügten schwarzen Kurve ausgeben, die den Farbtonwinkeln ab (x, y) folgt und endet, wenn die maximale Länge erreicht ist oder eine Bildgrenze erreicht.
Beginnen Sie die Kurve bei (x, y) und messen Sie dort den Farbtonwinkel. Gehen Sie eine Einheit (ein Pixel breit) in diese Richtung und beachten Sie, dass Ihre neue Position höchstwahrscheinlich keine Ganzzahlkoordinate ist . Markieren Sie einen anderen Punkt auf der Kurve und bewegen Sie sich erneut mit dem Farbton vom nächsten Pixel (mit etwas wie floor
oder round
, ich werde dies nicht genau überprüfen). Fahren Sie so fort, bis die Kurve außerhalb der Grenzen liegt oder die maximale Länge überschreitet. Zeichnen Sie zum Abschluss alle Kurvenpunkte als einzelne schwarze Pixel (verwenden Sie wieder die nächsten Pixel), die über dem Bild liegen, und geben Sie dieses neue Bild aus.
Der "Farbtonwinkel" ist nur der Farbton :
hue = atan2(sqrt(3) * (G - B), 2 * R - G - B)
Beachten Sie, dass für Graustufenwerte, die technisch gesehen keinen Farbton haben, 0 zurückgegeben wird, dies aber in Ordnung ist.
(Diese Formel verwendet atan2
die meisten eingebauten Mathematikbibliotheken. R, G, B liegen zwischen 0 und 1, nicht zwischen 0 und 255.)
- Sie können jedes gängige verlustfreie Bilddateiformat sowie alle Bildbibliotheken verwenden.
- Nehmen Sie Eingaben von stdin oder der Befehlszeile entgegen oder schreiben Sie eine Funktion mit Argumenten für den Namen der Bilddatei, x und y und die maximale Länge.
- Die maximale Länge und x und y sind immer nichtnegative ganze Zahlen. Sie können davon ausgehen, dass sich x und y im Bereich befinden.
- Speichern Sie das Ausgabebild unter einem Namen Ihrer Wahl oder zeigen Sie es einfach an.
- Ihre Implementierung muss nicht genau mit dem Snippet übereinstimmen. Ein paar Pixel an leicht unterschiedlichen Stellen aufgrund einer leicht unterschiedlichen Rundungs- / Berechnungsmethode sind in Ordnung. (In chaotischen Fällen kann dies dazu führen, dass Kurven sich stark voneinander unterscheiden. Solange sie jedoch optisch korrekt aussehen, ist dies in Ordnung.)
Wertung
Die kleinste Einsendung in Bytes gewinnt.
quelle
Antworten:
MATLAB, 136
Kopierte den größten Teil des Setups und ähnliches von @sanchises, aber ich benutze
streamline
es stattdessen , um den Pfad zu berechnen und zu zeichnen. Es wird jedoch ein Antialiasing für die Linie durchgeführt und die bilineare Interpolation verwendet, anstatt wie angegeben den nächsten Nachbarn.quelle
C mit SDL,
549516 BytesIch werde diese Party beginnen! Aus irgendeinem Grund hatte ich heute Abend Lust, Golf zu spielen. Was ihr tut, ist schwer ... Wenn es eine Sache gibt, die ich auf dieser Site nicht sehe, dann ist es SDL. Vielleicht habe ich gerade herausgefunden, warum. Dieses spezielle Snippet entspricht sowohl SDL2 als auch SDL1.2, ist aber grausam. Genannt wie
f("imagename.bmp", xcoord, ycoord, max_length);
. Es speichert eine Datei mit demselben Namen wie in den Argumenten angegeben. Die Ausgabe scheint dem Code-Snippet des OP sehr ähnlich zu sein, ist aber "unschärfer". Vielleicht versuche ich das etwas später zu beheben.Hier ist alles enträtselt:
Ich sollte beachten, dass darauf geachtet wurde, es plattformübergreifend zu gestalten - im Interesse der Ehrlichkeit fühlte es sich nicht gut an, es für meine Maschine fest zu codieren, selbst wenn eine erhebliche Anzahl von Bytes abgeschnitten wurde. Trotzdem bin ich der Meinung, dass ein paar Dinge hier überflüssig sind, und ich werde es später noch einmal anschauen.
BEARBEITEN-------
Immerhin handelt es sich um eine grafische Ausgabe ... Ich werde diese regelmäßig mit Bildern aktualisieren, die ich interessant finde.
f("HueTest1.bmp", 270, 40, 200);
f("HueTest2.bmp", 50, 50, 200);
f("HueTest3.bmp", 400, 400, 300);
quelle
Python,
203,172Beispielausgabe:
Mit anrufen
f("image.jpg", 400, 400, 300)
Ich habe viele Zeichen für den Import verschwendet, wenn jemand Vorschläge zur Verbesserung hat. Funktioniert möglicherweise nicht in 3.0
quelle
sqrt(3) -> 3**.5
? Bei den Importen fällt mir allerdings nichts ein.MATLAB
186,172Das Spiel ist auf ! Rufen Sie als
t('YourImage.ext',123,456,100')
für ein Bild eines beliebigen Typs auf, das MATLAB unterstützt, beginnend mit (x, y) = (123,456) und maximaler Länge 100. Anfangspositionen können nicht genau am rechten und unteren Rand sein (das würde mich zwei Bytes kosten) fang am rand an, benutze so etwas wiex=799.99
anfangen beix=800
. Die Indizierung beginnt bei 1 und nicht bei 0.Code:
Revisionen:
Ichbenutze immer noch,line
da dies der kürzeste Code ist, den ich zum Erzeugen eines Pixels kenne.Änderte die Farbe von schwarz nach blau, indem SieCo
auf expandierenColor
(was MATLAB automatisch macht)quelle
x=0
odery=0
möglicherweise gültige Positionen?function
Header gezählt. Spec erforderte keine nullbasierte oder einsbasierte Indizierung, daher verwende ich MATLABs einsbasierte, also nein,x=0
odery=0
ist in diesem Programm nicht gültig.x=X
undy=Y
gültig sein kann?Verarbeitung, 323 Zeichen
Mit Leerzeichen:
Ich bin mir sicher, dass ich dies weiter verkürzen könnte, aber es funktioniert vorerst.
quelle
JavaScript 414
quelle