Line drawing function/es

Esta página muestra cómo se puede crear funcionalidad avanzada con Python. En este ejercicio, construiremos una nueva herramienta que dibuja una línea. Esta herramienta puede ser vinculado a un comando FreeCAD, y ese comando se puede llamar desde cualquier elemento de la interfaz, tal como un elemento de menú o un botón de barra de herramientas.

El script principal
En primer lugar vamos a escribir un script (guión) que contenga todos nuestros funcionalidad. Despues, vamos a guardar esto en un archivo, e importarlo en FreeCAD, por lo que todas las clases y funciones que escriba estarán disponibles para FreeCAD. De modo que arranque su editor de texto favorito y escriba las siguientes líneas:

import FreeCADGui, Part from pivy.coin import * class line: "this class will create a line after the user clicked 2 points on the screen" def __init__(self): self.view = FreeCADGui.ActiveDocument.ActiveView self.stack = [] self.callback = self.view.addEventCallbackPivy(SoMouseButtonEvent.getClassTypeId,self.getpoint) def getpoint(self,event_cb): event = event_cb.getEvent if event.getState == SoMouseButtonEvent.DOWN: pos = event.getPosition point = self.view.getPoint(pos[0],pos[1]) self.stack.append(point) if len(self.stack) == 2: l = Part.Line(self.stack[0],self.stack[1]) shape = l.toShape Part.show(shape) self.view.removeEventCallbackPivy(SoMouseButtonEvent.getClassTypeId,self.callback)

Explicación detallada
import Part, FreeCADGui from pivy.coin import *

En Python, cuando desee utilizar las funciones de otro módulo, tiene que importarlo. En nuestro caso, vamos a necesitar las funciones del Part Module, para la creación de la línea, y del módulo GUI (FreeCADGui), para acceder a la vista 3D. También necesitamos el contenido completo de la librería de Coin, para que podamos utilizar directamente todos los objetos Coin, como SoMouseButtonEvent, etc ..

class line:

Aquí definimos nuestra clase principal. ¿Por qué utilizar una clase y no una función? La razón es que necesitamos que nuestra herramienta se mantenga "viva" mientras esperamos a que el usuario haga clic en la pantalla. Una función termina cuando su tarea se ha hecho, pero un objeto (una clase se define como un objeto) se mantiene vivo hasta que se destruye.

"Esta clase va a crear una línea después de que el usuario hace clic en 2 puntos en la pantalla"

En Python, cada clase o función puede tener una cadena de descripción. Esto es particularmente útil en FreeCAD, porque cuando vas a llamar a esa clase en el intérprete, la cadena de descripción se mostrará como una nota.

def __init__(self):

Las clases en Python siempre pueden contener una función __init__, que se ejecuta cuando la clase es llamada para crear un objeto. Por lo tanto, vamos a poner aquí todo lo que queremos que ocurra cuando la herramienta de línea comienza.

self.view = FreeCADGui.ActiveDocument.ActiveView

En una clase, por lo general querrá incluir self. antes de un nombre de variable, para que sea fácilmente accesible a todas las funciones dentro y fuera de esa clase. Aquí, vamos a utilizar self.view para acceder y manipular la vista 3D activa.

self.stack = []

Aquí creamos una lista vacía que contendrá los puntos 3D enviados por la función getpoint.

self.callback = self.view.addEventCallbackPivy(SoMouseButtonEvent.getClassTypeId,self.getpoint)

Aquí viene la parte importante: Dado que en realidad se trata de una escena Coin3d, FreeCAD utiliza el mecanismo de devolución de llamada de Coin, que permite que una función sea llamada cada vez que sucede un determinado evento de escena. En nuestro caso, estamos creando una devolución de llamada para eventos SoMouseButtonEvent, y lo conduciremos a la función getpoint. Ahora, cada vez que un botón del ratón sea pulsado o soltado, la función getpoint será ejecutada.

Tenga en cuenta que también hay una alternativa a addEventCallbackPivy, llamada addEventCallback, que dispensa del uso de pivy. Pero como pivy es una forma muy eficaz y natural para acceder a cualquier parte de la escena de Coin, es mucho mejor usarlo tanto como se pueda!

def getpoint(self,event_cb):

Ahora definimos la función getpoint, que se ejecutará cuando el botón del ratón se pulsa en una vista 3D. Esta función recibe un argumento, que llamaremos event_cb. A partir de este evento de devolución de llamada podemos tener acceso al objeto de evento, que contiene varias piezas de información (mas info here).

if event.getState == SoMouseButtonEvent.DOWN:

La función getpoint se llamará cuando un botón del ratón sea pulsado o soltado. Pero queremos escoger un punto 3D sólo cuando se presiona (de lo contrario obtendríamos dos puntos 3D muy cerca uno del otro). Por lo tanto, debe comprobar eso aquí.

pos = event.getPosition

Aquí obtenemos las coordenadas de pantalla del cursor del ratón

point = self.view.getPoint(pos[0],pos[1])

Esta función nos da un vector FreeCAD (x,y,z) que contiene el punto 3D que se encuentra en el plano focal, justo debajo de nuestro cursor de ratón. Si usted está en vista de cámara, imagine un rayo proveniente de la cámara, pasando por el cursor del ratón, y alcanzando el plano focal. Ahí está nuestro punto 3D. Si estamos en vista ortogonal, el rayo es paralelo a la dirección de la vista.

self.stack.append(point)

Añadimos nuestro nuevo punto a la pila

if len(self.stack) == 2:

¿Tenemos ya suficientes puntos? si es así, entonces vamos a trazar la línea!

l = Part.Line(self.stack[0],self.stack[1])

Aquí se utiliza la función line de Part Module que crea una línea a partir de dos vectores FreeCAD. Todo lo que creamos y modificamos dentro del módulo Part, se queda en el módulo Part. Así, hasta ahora, hemos creado un elemento de línea. No está ligado a un objeto de nuestro documento activo, por lo que no aparece nada en la pantalla.

shape = l.toShape

El documento FreeCAD sólo puede aceptar formas desde el módulo Parte. Las formas son el tipo más genérico del módulo Parte (pieza). Por lo tanto, debemos convertir nuestra línea en una forma antes de añadirla al documento.

Part.show(shape)

El módulo Parte (de pieza) tiene una función, show, que es muy útil ya que crea un nuevo objeto en el documento y le conecta a una forma. También podría haber creado primero un nuevo objeto en el documento, y a continuación vincularle a la forma manualmente.

self.view.removeEventCallbackPivy(SoMouseButtonEvent.getClassTypeId,self.callback)

Como ya hemos terminado con nuestra línea, vamos a quitar el mecanismo de devolución de llamada, que consume preciosos ciclos de CPU.

Pruebas y Uso del script
Ahora, vamos a guardar nuestro script (guión o secuencia de comandos) en un lugar donde el intérprete de Python FreeCAD lo encontrará. Cuando importa los módulos, el intérprete mirará en los siguientes lugares: las rutas de instalación de Python, el directorio bin de FreeCAD, y todos los directorios de módulos FreeCAD. Por lo tanto, la mejor solución es crear un nuevo directorio en una de los FreeCAD Mod directories, y salvar nuestro script en él. Por ejemplo, vamos a hacer un directorio "MyScripts", y salvamos nuestro script como "exercise.py".

Ahora, todo está listo, vamos a empezar FreeCAD, cree un nuevo documento, y, en el intérprete de Python, ejecute:

import exercise

Si no aparece ningún mensaje de error, eso significa que nuestro script de ejercicio se ha cargado. Ahora puede comprobar su contenido con:

dir(exercise)

El comando dir es un comando integrado de python que muestra el contenido de un módulo. Podemos ver que nuestra clase line está ahí, esperandonos. Ahora vamos a probarlo:

exercise.line

A continuación, haga clic dos veces en la vista 3D, y .... ¡bingo!, ¡aquí está nuestra línea! Para hacerlo de nuevo, simplemente escriba exercise.line otra vez, y otra vez, y otra vez ... Se siente feliz, ¿no?

Incluyendo el script en la interfaz de FreeCAD
Ahora, para que nuestra nuevo herramienta línea sea realmente buena, debe tener un botón en la interfaz, para que no sea necesario escribir todas esas cosas cada vez. La forma más fácil es transformar nuestro nuevo directorio MyScripts en un completo entorno FreeCAD. Es fácil, todo lo que se necesita es poner un archivo llamado InitGui.py dentro de su directorio MyScripts. El InitGui.py contendrá las instrucciones para crear un nuevo entorno (workbench), y le añadimos nuestra nueva herramienta. Además, también habrá que transformar un poco nuestro código de ejercicio, para que la herramienta line sea reconocida como un comando de oficial de FreeCAD. Comencemos por crear un archivo InitGui.py, y escribir el siguiente código en él:

class MyWorkbench (Workbench): MenuText = "MyScripts" def Initialize(self): import exercise commandslist = ["line"] self.appendToolbar("My Scripts",commandslist) Gui.addWorkbench(MyWorkbench)

A estas alturas, usted ya debe entender el script anterior por sí mismo, supongo: Creamos una nueva clase que llamamos MyWorkbench, le damos un título (MenuText), y definimos una función initialize que se ejecutará cuando el entorno se cargue en FreeCAD. En esa función, se carga el contenido de nuestro archivo de ejercicio, y los comandos FreeCAD que encuentra dentro se anexan en una lista de comandos. A continuación, hacemos una barra de herramientas llamada "Mi Scripts" y le asignamos nuestra lista de comandos. Finalmente, por supuesto, sólo tenemos una herramienta, por lo que nuestra lista de comandos contiene un solo elemento. Ahora, una vez que nuestro entorno está listo, lo añadimos a la interfaz principal.

Pero esto todavía no funciona, porque un comando FreeCAD debe estar formateado de una determinada manera para poder funcionar. Así que tendremos que transformar un poco nuestra herramienta line. Nuestro nuevo script exercise.py tendrá despues este aspecto:

import FreeCADGui, Part from pivy.coin import * class line: "this class will create a line after the user clicked 2 points on the screen" def Activated(self): self.view = FreeCADGui.ActiveDocument.ActiveView self.stack = [] self.callback = self.view.addEventCallbackPivy(SoMouseButtonEvent.getClassTypeId,self.getpoint) def getpoint(self,event_cb): event = event_cb.getEvent if event.getState == SoMouseButtonEvent.DOWN: pos = event.getPosition point = self.view.getPoint(pos[0],pos[1]) self.stack.append(point) if len(self.stack) == 2: l = Part.Line(self.stack[0],self.stack[1]) shape = l.toShape Part.show(shape) self.view.removeEventCallbackPivy(SoMouseButtonEvent.getClassTypeId,self.callback) def GetResources(self): return {'Pixmap' : 'path_to_an_icon/line_icon.png', 'MenuText': 'Line', 'ToolTip': 'Creates a line by clicking 2 points on the screen'} FreeCADGui.addCommand('line', line)

Lo que hicimos aquí es transformar nuestra función __init__ en una función Activated, porque cuando se ejecutan comandos FreeCAD, automáticamente ejecutan la función Activated. También hemos añadido una función GetResources, que informa a FreeCAD donde puede encontrar un icono de la herramienta, y cual será el nombre y la descripción de nuestra herramienta. Cualquier imagen jpg, png o svg funcionará como un icono. Puede ser de cualquier tamaño, pero lo mejor es utilizar un tamaño que esté cerca del aspecto final, como 16x16, 24x24 o 32x32. A continuación, añadimos la clase line como un comando FreeCAD oficial con el método addCommand.

Eso es todo, ahora sólo hay que reiniciar FreeCAD y tendremos un agradable entorno nuevo con una nueva herramienta line de nuestra marca!

¿Quiere más?
Si le gustó este ejercicio, ¿por qué no tratar de mejorar esta pequeña herramienta? Hay muchas cosas que se pueden hacer, como por ejemplo:
 * Agregar asistencia para los usuarios: hasta ahora hemos hecho una herramienta muy burda, el usuario podría verse un poco perdido cuando la usa. Se podría añadir alguna información, diciéndole qué hacer a continuación. Por ejemplo, usted podría sacar mensajes en la consola FreeCAD. ¡Eche un vistazo en el módulo de FreeCAD.Console
 * Añadir la posibilidad de teclear de forma manual las coordenadas de los puntos 3D. Mire en la función input de Python, por ejemplo
 * Añadir la posibilidad de incluir más de 2 puntos
 * Añadir eventos para otras cosas: Ahora sólo comprobamos eventos de botón del ratón, ¿que tal si también hace algo cuando el ratón se mueva, como mostrar las coordenadas actuales?
 * Dar un nombre al objeto creado

No dude en escribir sus preguntas o ideas en talk page!