Rasterung von PolygonenDie Rasterung von Polygonen und aneinandergereihten Liniensegmenten (Polygonzügen) ist eine Aufgabe der Computergrafik. Das Rastern von Polygonzügen basiert auf der Rasterung von Linien, erfordert jedoch bei dicker Strichbreite zusätzlichen Aufwand. Dem Rastern von gefüllten Polygonen kommt in der 3D-Computergrafik eine große Rolle zu, da 3D-Szenen gerendert werden können, indem man die auf die Bildebene projizierten Polygone farbig füllt. Verbinden von LiniensegmentenBeim Rastern dicker Liniensegmente muss entschieden werden, wie sie miteinander verbunden werden. Dicke Linien als Rechtecke zu betrachten, liefert unschöne Ergebnisse. Besser ist es, Linien mit runden Enden zu zeichnen. Eine andere Möglichkeit ist Mitering („Gehrung“), bei der die Linienenden schräg gezeichnet werden, sodass sie aneinander angrenzen. Bei sehr spitzen Winkeln ragen die Linienenden zu sehr über die eigentliche Polygonecke heraus, weshalb sie besser abgeschnitten werden. ArtefakteBei der Rasterung von Polygonen muss entschieden werden, wie deren Koordinaten interpretiert werden. Wenn beispielsweise ein Rechteck mit den Endpunktkoordinaten (1, 1) und (5, 4) gerastert wird, so werden im Normalfall 5×4 Pixel eingefärbt, obwohl das Rechteck nur 4×3 Einheiten groß ist. Dieser unerwünschte Effekt ist eine Konsequenz der endlichen Bildauflösung. Wenn nebeneinanderliegende Polygone gezeichnet werden, so führt das dazu, dass einige Pixel mehrfach eingefärbt werden. Eine Möglichkeit, dieses Problem zumindest bei ganzzahligen Koordinaten zu umgehen, besteht darin, sie bei der Rasterung um einen halben Pixelabstand nach links und nach unten zu verschieben, sodass in Wirklichkeit das Rechteck mit den Koordinaten (0,5, 0,5) und (4,5, 3,5) gerastert wird (siehe Bild). Dadurch wird vermieden, dass bei nebeneinanderliegenden Polygonen Kanten doppelt eingefärbt werden, was bei bestimmten Rasteroperationen wie XOR unerwünschte Resultate hervorrufen würde. Eine alternative Methode, die Gleitkommazahlen vermeidet, ist, das jeweils letzte Pixel einer Zeile oder einer Spalte nicht zu rastern. Dabei wird in Kauf genommen, dass das Polygon etwas verschoben erscheint. Auch sehr spitzwinklige Polygonecken können dazu führen, dass Pixel jeweils von mehreren Segmenten eingefärbt werden. Ein weiteres Beispiel für Artefakte sind Slivers, Polygonteile, die so dünn sind, dass sie keine Pixel einschließen und bei denen einige Rasteralgorithmen nur einzelne oder gar keine Pixel zeichnen. Wenn ein gerasterter Winkel durch eine weitere Linie halbiert werden soll, etwa um Pfeile darzustellen, so ist es im Allgemeinen nicht möglich, ein exakt symmetrisches Resultat zu erreichen.[1] Aneinanderliegende DreieckePolygone mit mehr als drei Ecken als Teil von Polygonnetzen werden vor der Rasterung oft in Dreiecke umgewandelt. Dabei teilen sich sehr viele Dreiecke ihre Kanten. Wenn diese aneinanderliegenden Dreiecke unterschiedlich gefärbt sind und jede Kante als einfarbige Linie gerastert wird, so hängt das Erscheinungsbild von der Reihenfolge ab, in der die Dreiecke gerastert werden. Um dieses Problem zu umgehen, färbt man ein Pixel dann und nur dann ein, wenn es innerhalb des zu zeichnenden Dreiecks liegt. Dazu können baryzentrische Koordinaten verwendet werden. Bezogen auf ein Dreieck kann jeder Punkt der Ebene durch die Koordinaten beschrieben werden. Ein Punkt oder Pixel befindet sich genau dann strikt innerhalb dieses Dreiecks, wenn sich jede dieser Koordinaten im Intervall befindet. Diese Methode ist auch für nicht-ganzzahlige Eckpunktkoordinaten gültig. Einen Sonderfall stellen Pixel dar, die sich genau auf einer Kante befinden und somit nicht trivial einem der beiden anliegenden Dreiecke zugewiesen werden können. Für diese Fälle wählt man einen außerhalb des Bildes liegenden Referenzpunkt und wählt die Farbe desjenigen Dreiecks, dessen nicht auf der Kante befindlicher Eckpunkt näher an diesem Punkt liegt. Diese Methode funktioniert immer dann, wenn der Referenzpunkt nicht auf der durch die Kante verlaufenden Geraden liegt. FüllungAm einfachsten werden gefüllte Dreiecke und damit Polygone gerastert, indem für jedes Pixel des Bildes der oben beschriebene Test angewendet wird: Pixel werden nur dann eingefärbt, wenn sie innerhalb eines Dreiecks liegen. Eine etwas effizientere Methode testet nur diejenigen Pixel innerhalb eines Rechtecks, welches das zu rasternde Dreieck einschließt. Neben diesen einfachen Methoden sind jedoch schnellere Verfahren entwickelt worden, die im Folgenden beschrieben werden. Kantenlisten-AlgorithmenGrundprinzipKantenlisten-Algorithmen bestimmen für jede Kante des Polygons die Schnittpunkte mit den Bildzeilen. Diese können direkt berechnet werden, oder es können Algorithmen zur Rasterung von Linien verwendet werden. Horizontale Kanten werden ignoriert. Die so ermittelten Schnittpunkte werden in einer Liste abgelegt. Um wie weiter oben beschrieben zu vermeiden, dass zu viele Pixel an den Kanten eingefärbt werden, werden in Wirklichkeit die Schnittpunkte mit genau zwischen zwei Bildzeilen verlaufenden Geraden berechnet. Für das rechts im Bild dargestellte Beispielpolygon werden somit folgende Schnittpunkte in der Liste abgelegt:
Diese Schnittpunkte werden nun nach y-Koordinaten absteigend sortiert. Unter Werten mit gleichen y-Koordinaten wird aufsteigend nach x-Koordinaten sortiert. Nach der Sortierung sieht die Liste folgendermaßen aus: (1; 6,5) (1,5; 6,5) (1; 5,5) (2,5; 5,5) (7,5; 5,5) (8; 5,5) (1; 4,5) (3,5; 4,5) (6,5; 4,5) (8; 4,5) (1; 3,5) (4,5; 3,5) (5,5; 3,5) (8; 3,5) (1; 2,5) (8; 2,5) (1; 1,5) (8; 1,5) In dieser Liste gibt es immer eine gerade Anzahl von Werten mit gleicher y-Koordinate. Das Polygon wird gezeichnet, indem nacheinander Punktpaare aus der Liste betrachtet werden. Sie sind stets von der Form (x1; y) (x2; y); beide Punkte haben also die gleiche y-Koordinate. Es werden nun alle Pixel der entsprechenden Bildzeile gezeichnet, deren x-Koordinate sich im Intervall befindet. Das erste Punktpaar ist (1; 6,5) (1,5; 6,5). Das einzige Pixel, das die eben genannte Bedingung erfüllt, ist hier (1; 6). Anschließend wird das nächste Punktpaar, (1; 5,5) (2,5; 5,5), betrachtet. Hier gibt es zwei Pixel, die eingefärbt werden, nämlich (1; 5) und (2; 5). Das Polygon ist fertig gerastert, wenn alle Punktpaare der sortierten Liste abgearbeitet wurden. Aktive KantenlisteDie Vorausberechnung der Schnittpunkte von Polygonkanten und zwischen den Bildzeilen verlaufenden Geraden ist unnötig zeitaufwändig und kann erheblichen Speicherplatz erfordern. Mit einer so genannten aktiven Kantenliste lässt sich die Berechnung der Schnittpunkte inkrementell durchführen und der Speicherplatz reduzieren. Dieses Verfahren wird gelegentlich Scanline-Algorithmus genannt;[2] allerdings werden mit Scanline-Algorithmen auch darauf aufbauende Verfahren bezeichnet, um im Rahmen der 3D-Computergrafik aus Polygonen aufgebaute Szenen Zeile für Zeile zu rastern. Bei diesem Algorithmus werden für jede Polygonkante nicht die Schnittpunkte mit allen Geraden, sondern nur mit der Geraden mit der größten y-Koordinate, die die Kante schneidet, ermittelt. Zusätzlich zur x-Koordinate des Schnittpunkts werden folgende Daten ermittelt:
Die Daten werden in einer Tabelle gespeichert, deren Einträge nach der Bildzeile sortiert sind. Für das Beispielpolygon ergibt sich folgende Tabelle:
Die Grundidee des Algorithmus besteht darin, von diesen vorberechneten Daten auszugehen und mit Hilfe der Δx-Werte die Koordinaten der anderen Schnittpunkte fortlaufend zu berechnen. Dabei wird von der höchsten Bildzeile ausgegangen und schrittweise zur niedrigeren Bildzeile gewechselt. Eine aktive Kantenliste speichert die Kanten, die die zur Bildzeile gehörende Gerade schneiden, sowie für jede Kante die aktuellen x-, Δx- und ny-Werte. Zu Beginn ist die aktive Kantenliste leer. Ausgegangen wird von der höchsten Bildzeile. Für jede Bildzeile wird in der vorberechneten Tabelle gesucht, ob sie Kanten enthält, die noch nicht in der aktiven Kantenliste enthalten sind, und wenn ja, werden die entsprechenden Daten in die aktive Kantenliste kopiert. Nun werden die x-Werte aller in der aktiven Kantenliste befindlichen Kanten aufsteigend sortiert. Die resultierenden Punktpaare werden wie im grundlegenden Kantenlisten-Algorithmus dazu verwendet, Pixel einzufärben. Anschließend werden die ny-Werte in der aktiven Kantenliste um 1 erniedrigt; fällt ein Wert unter 0, so wird die entsprechende Kante aus der aktiven Kantenliste entfernt. Schließlich werden zu allen x-Werten in der aktiven Kantenliste Δx addiert, und die Prozedur wird mit der nächsten, niedrigeren Bildzeile wiederholt. Die Rasterung ist fertig, sobald alle Bildzeilen abgearbeitet wurden. Beim Beispielpolygon verändert sich die aktive Kantenliste in den ersten vier Bildzeilen wie folgt:
Edge-fill-AlgorithmusDer größte Nachteil der Kantenlisten-Algorithmen ist der Aufwand zur Sortierung und Manipulation der Listen. Der sehr einfache Edge-fill-Algorithmus[3] kommt ohne diesen Aufwand aus. Beim Edge-fill-Algorithmus werden für jede Bildzeile, die bei eine Polygonkante schneidet, alle Pixel dieser Bildzeile mit einer x-Koordinate strikt größer als invertiert. „Invertierung“ bedeutet hier, dass eingefärbte Pixel in den Ausgangszustand zurückgesetzt werden und umgekehrt. Die Reihenfolge, in der die Polygonkanten abgearbeitet werden, ist beliebig. Wenn der Algorithmus die Kanten des Beispielpolygons gegen den Uhrzeigersinn abarbeitet, so ergeben sich folgende Schritte: Das gerasterte Polygon unterscheidet sich von dem der Kantenlisten-Algorithmen, denn drei Pixel werden nicht eingefärbt. Der Nachteil des Algorithmus ist, dass viele Pixel mehrmals geändert werden müssen. Fence-fill-AlgorithmusDer Fence-fill-Algorithmus[4] ist eine Weiterentwicklung des Edge-fill-Algorithmus, der die Zahl der nötigen Pixelinvertierungen verringert. Dabei wird ein Zaun (engl. “fence”), eine durch den Schnittpunkt zweier Polygonkanten verlaufende vertikale Gerade, genutzt.
Dadurch ergeben sich folgende Schritte: Edge-flag-AlgorithmusDer Edge-flag-Algorithmus[5] besteht aus zwei Schritten:
Für jede Polygonkante linie, des Polygons punkt1 = linie.punkt1 punkt2 = linie.punkt2 wenn (punkt1.y > punkt2.y) dann tausche punkt1 mit punkt2 Für jede Bildzeile y, des Bildes steigung_inv = (punkt1.x - punkt2.x) / (punkt1.y - punkt2.y) schnittpunkt_y = y + 0.5 schnittpunkt_x = punkt1.x + steigung_inv * (schnittpunkt_y - punkt1.y) x = abrunden(schnittpunkt_x) wenn (x + 0,5) <= schnittpunkt_x dann x=x+ 1 //- Randpunkt des Schnittpunktes mit der Bildzeile umkehren (einfärben oder zurücksetzen) Pixel (x,y) = ! Pixel (x,y);
Für jede Bildzeile y, die das Polygon schneidet Innerhalb = Falsch Für jedes x von links bis rechts Wenn Pixel (x, y) eingefärbt ist Innerhalb negieren Wenn Innerhalb Pixel (x, y) einfärben ansonsten Pixel (x, y) auf Hintergrundfarbe zurücksetzen Als Softwareimplementierung sind der Kantenlisten-Algorithmus und der Edge-flag-Algorithmus vergleichbar schnell,[6] implementiert in Hardware ist letzterer jedoch erheblich schneller. Siehe auchLiteratur
Einzelnachweise
|