{"id":599,"date":"2014-09-29T21:03:07","date_gmt":"2014-09-29T18:03:07","guid":{"rendered":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/?page_id=599"},"modified":"2014-10-01T16:11:14","modified_gmt":"2014-10-01T13:11:14","slug":"desenhar-formas-geometricas","status":"publish","type":"page","link":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/opencv2-python27\/capitulo1-basico\/desenhar-formas-geometricas\/","title":{"rendered":"1.3 &#8211; Desenhar formas geom\u00e9tricas"},"content":{"rendered":"<p>As formas geom\u00e9tricas b\u00e1sicas que poder\u00e3o ser desenhadas utilizando a OpenCV s\u00e3o a linha, o ret\u00e2ngulo e o c\u00edrculo. O princ\u00edpio b\u00e1sico de aplica\u00e7\u00e3o \u00e9 o mesmo, indicando a forma, a posi\u00e7\u00e3o, a cor e a espessura do tra\u00e7o.<\/p>\n<p>Nesse tutorial, al\u00e9m da OpenCV tamb\u00e9m \u00e9 utilizada a biblioteca Numpy para a manipula\u00e7\u00e3o inicial da imagem. A biblioteca Numpy pode ser instalada com instru\u00e7\u00f5es a partir do <a title=\"Link para o website do Numpy\" href=\"http:\/\/www.numpy.org\" target=\"_blank\">website do projeto<\/a>.<\/p>\n<p><strong>Criando o fundo da imagem<\/strong><\/p>\n<p>A aplica\u00e7\u00e3o das formas pode ser feita em uma imagem carregada ou capturada de um v\u00eddeo ou webcam, mas nesse tutorial criaremos uma imagem em branco e adicionaremos as formas nessa imagem. Criar a imagem em branco significar\u00e1 criar uma matriz colocando cada pixel da matriz com todas as cores. Para facilitar essa manipula\u00e7\u00e3o inicial de matriz, ser\u00e1 utilizada a biblioteca Numpy, sendo necess\u00e1rio que ela esteja instalada junto ao Python.<\/p>\n<p>Atrav\u00e9s da instru\u00e7\u00e3o da linha 4, \u00e9 solicitado ao Numpy que crie uma matriz\u00a0tridimensional, tendo 300 linhas, 400 colunas e a terceira dimens\u00e3o para manter os 3 canais para as cores RGB. O m\u00e9todo <em>ones()<\/em>\u00a0da Numpy cria uma matriz toda preenchida com o n\u00famero 1, sendo ent\u00e3o necess\u00e1rio multiplicar os valores dessa matriz por 255 para obter a cor branca. A linha 5 mostra a imagem em uma janela e na linha 6 \u00e9 esperado alguma tecla ser pressionada para os passos seguintes.<\/p>\n<pre class=\"lang:python decode:true\">import cv2\r\nimport numpy as np   \r\n    \r\ncanvas = np.ones((300, 400, 3)) * 255 #imagem 400x300, com fundo branco e 3 canais para as cores\r\ncv2.imshow(\"Canvas\", canvas)\r\ncv2.waitKey(0)<\/pre>\n<p><strong>Desenhando linhas<\/strong><\/p>\n<p>Uma linha \u00e9 desenhada atrav\u00e9s do m\u00e9todo <em>line()<\/em> da OpenCV (linha 9). Esse m\u00e9todo espera como primeiro par\u00e2metro em qual imagem ser\u00e1 desenhada a linha. O segundo par\u00e2metro \u00e9 a coordenada onde a linha ser\u00e1 iniciada, enquanto o terceiro par\u00e2metro \u00e9 a coordenada onde a linha finalizar\u00e1.\u00a0Ser\u00e1 tra\u00e7ada uma linha reta entre a coordenada inicial e final. O quarto par\u00e2metro \u00e9 a especifica\u00e7\u00e3o de uma cor para a linha. No c\u00f3digo mostrado abaixo, a cor foi definida em uma vari\u00e1vel para ser possivelmente reaproveitada (embora n\u00e3o tenha sido no c\u00f3digo). Observe que as coordenadas e a cor s\u00e3o <em>arrays<\/em>.<\/p>\n<p>As linhas 10 e 11 s\u00e3o respons\u00e1veis apenas por exibir a imagem na janela aberta anteriormente e esperar ser pressionada alguma tecla. Observe que a janela anterior e a solicitada na linha 10 possuem o mesmo nome e por isso a imagem ser\u00e1 apenas atualizada na janela. Se nomes diferentes fossem utilizados, uma nova janela seria criada.<\/p>\n<p>A primeira linha especificada corta a imagem na diagonal, saindo do canto superior esquerdo (coordenada 0, 0) e indo at\u00e9 o canto inferior direito (coordenada 300, 400 do plano cartesiano &#8211; coordenada 400, 300 da OpenCV).<\/p>\n<p>Na linha 15 \u00e9 criada uma outra linha, sendo essa uma linha que cortar\u00e1 a imagem ao meio e na horizontal. O m\u00e9todo line() nessa instru\u00e7\u00e3o recebeu um par\u00e2metro adicional que \u00e9 o n\u00famero de pixels para a espessura da linha. O padr\u00e3o \u00e9 fazer a forma com 1 pixel\u00a0e\u00a0sendo assim, a segunda linha ser\u00e1 mais espessa do que a primeira. A segunda linha tamb\u00e9m recebeu a cor verde, definida atrav\u00e9s da linha 14.<\/p>\n<pre class=\"start-line:7 lang:python decode:true\">#desenha a linha diagonal    \r\nazul = (255, 0, 0)\r\ncv2.line(canvas, (0, 0), (400, 300), azul)\r\ncv2.imshow(\"Canvas\", canvas)\r\ncv2.waitKey(0)\r\n\r\n#desenha a linha vertical\r\nverde = (0, 255, 0)\r\ncv2.line(canvas, (200, 0), (200, 300), verde, 3)\r\ncv2.imshow(\"Canvas\", canvas)\r\ncv2.waitKey(0)<\/pre>\n<p><strong>Desenhando ret\u00e2ngulos<\/strong><\/p>\n<p>O m\u00e9todo <em>rectangle()<\/em> da OpenCV \u00e9 respons\u00e1vel por desenhar um ret\u00e2ngulo e seus par\u00e2metros s\u00e3o similares ao do m\u00e9todo <em>line()<\/em>, sendo necess\u00e1rio informar a imagem que receber\u00e1 a forma, as coordenadas inicial e final, uma cor e possibilidade de um valor para a espessura da linha (que ser\u00e1 o\u00a0contorno do ret\u00e2ngulo). Um ret\u00e2ngulo ser\u00e1 formado entre as coordenadas inicial e final. A linha 19 \u00e9 a instru\u00e7\u00e3o respons\u00e1vel por desenhar o ret\u00e2ngulo verde visto na figura ao final do tutorial.<\/p>\n<p>Observe que na linha 25 foi informado o valor -1 como valor da espessura da imagem. Esse valor negativo\u00a0gerar\u00e1 um ret\u00e2ngulo preenchido e n\u00e3o apenas contornado. A instru\u00e7\u00e3o dessa linha \u00e9 a respons\u00e1vel por gerar o ret\u00e2ngulo vermelho da figura.<\/p>\n<pre class=\"start-line:18 lang:python decode:true\">#desenha o ret\u00e2ngulo com borda verde    \r\ncv2.rectangle(canvas, (10, 70), (90, 190), verde)\r\ncv2.imshow(\"Canvas\", canvas)\r\ncv2.waitKey(0)\r\n\r\n#desenha o ret\u00e2ngulo todo vermelho\r\nvermelho = (0, 0, 255)\r\ncv2.rectangle(canvas, (250, 50), (300, 125), vermelho, -1)\r\ncv2.imshow(\"Canvas\", canvas)\r\ncv2.waitKey(0)<\/pre>\n<p><strong>Desenhando c\u00edrculos<\/strong><\/p>\n<p>Por fim \u00e9 mostrado como desenhar um c\u00edrculo na imagem. O m\u00e9todo circle() \u00e9 o m\u00e9todo da OpenCV encarregado dessa tarefa. Esse m\u00e9todo espera como primeiro par\u00e2metro a imagem que receber\u00e1 a forma e como segundo par\u00e2metro uma coordenada, que ser\u00e1 o centro do c\u00edrculo. O terceiro par\u00e2metro \u00e9 o di\u00e2metro que o c\u00edrculo dever\u00e1 ter. O quarto par\u00e2metro a ser informado \u00e9 a cor, podendo tamb\u00e9m receber um quinto par\u00e2metro com a espessura da linha, ou o valor -1 para formar um c\u00edrculo todo preenchido. A linha 30\u00a0do c\u00f3digo \u00e9 a respons\u00e1vel por desenhar o c\u00edrculo visto na figura a seguir.<\/p>\n<pre class=\"start-line:28 lang:python decode:true\">#desenha o c\u00edrculo    \r\npreto = (0, 0, 0)\r\ncv2.circle(canvas, (130, 230), 50, preto)\r\ncv2.imshow(\"Canvas\", canvas)\r\ncv2.waitKey(0)<\/pre>\n<p style=\"text-align: center;\"><a href=\"http:\/\/www.galirows.com.br\/meublog\/wp-content\/uploads\/2014\/09\/canvas.png\"><img loading=\"lazy\" decoding=\"async\" class=\"size-full wp-image-599 aligncenter\" src=\"http:\/\/www.galirows.com.br\/meublog\/wp-content\/uploads\/2014\/09\/canvas.png\" alt=\"canvas\" width=\"399\" height=\"299\" \/><\/a><strong>Figura gerada<\/strong><\/p>\n<p style=\"text-align: left;\">Veja que como foi utilizada a instru\u00e7\u00e3o <em>cv2.waitKey()<\/em> ao final de cada forma desenhada, a cada tecla pressionada ser\u00e1 mostrado a adi\u00e7\u00e3o de um novo elemento gr\u00e1fico definido.<\/p>\n","protected":false},"excerpt":{"rendered":"<p>As formas geom\u00e9tricas b\u00e1sicas que poder\u00e3o ser desenhadas utilizando a OpenCV s\u00e3o a linha, o ret\u00e2ngulo e o c\u00edrculo. O princ\u00edpio b\u00e1sico de aplica\u00e7\u00e3o \u00e9 o mesmo, indicando a forma, a posi\u00e7\u00e3o, a cor e a espessura do tra\u00e7o. Nesse &hellip; <a href=\"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/opencv2-python27\/capitulo1-basico\/desenhar-formas-geometricas\/\">Continue lendo <span class=\"meta-nav\">&rarr;<\/span><\/a><\/p>\n","protected":false},"author":1,"featured_media":650,"parent":609,"menu_order":4,"comment_status":"open","ping_status":"closed","template":"onecolumn-page.php","meta":{"footnotes":""},"class_list":["post-599","page","type-page","status-publish","has-post-thumbnail","hentry"],"aioseo_notices":[],"_links":{"self":[{"href":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/wp-json\/wp\/v2\/pages\/599","targetHints":{"allow":["GET"]}}],"collection":[{"href":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/wp-json\/wp\/v2\/pages"}],"about":[{"href":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/wp-json\/wp\/v2\/types\/page"}],"author":[{"embeddable":true,"href":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/wp-json\/wp\/v2\/comments?post=599"}],"version-history":[{"count":4,"href":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/wp-json\/wp\/v2\/pages\/599\/revisions"}],"predecessor-version":[{"id":633,"href":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/wp-json\/wp\/v2\/pages\/599\/revisions\/633"}],"up":[{"embeddable":true,"href":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/wp-json\/wp\/v2\/pages\/609"}],"wp:featuredmedia":[{"embeddable":true,"href":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/wp-json\/wp\/v2\/media\/650"}],"wp:attachment":[{"href":"http:\/\/www.galirows.com.br\/meublog\/opencv-python\/wp-json\/wp\/v2\/media?parent=599"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}