viernes, 18 de noviembre de 2011

C#: Ficheros y rutas de trabajo temporales en Windows.

Ésto lo escribo más como un recordatorio para mi mismo que otra cosa.

Para obtener el directorio de trabajo temporal en Windows, puede usarse la siguiente función:

Path.GetTempPath() 

Esta función crea un fichero vacío con un nombre único , y te devuelve su nombre:

Path.GetTempFileName()

sábado, 6 de agosto de 2011

Jazztel, no quiero tu puto ADSL

Como estoy pensando en cambiar de ADSL, y vivo en un pueblo, hace unos meses se me ocurrió comprobar la cobertura que tenía en distintas compañías. Miré en las páginas web de ONO , Jazztel, y no sé si alguna más.

Sabía que, al introducir mi número de teléfono en una página web, iba a tener alguna llamada comercial. En ONO, por ejemplo, nos llamaron una vez para ofrecerme sus servicios. Bien, es razonable. Pero lo que no esperaba es la pesadilla que ha venido después.

Durante MESES, hemos recibido de una a tres llamadas diarias por parte de Jazztel para ofrecernos su ADSL. Hemos probado de todo, desde decir que no educadamente, hasta explicarles que ya nunca contrataremos nada en su compañía, hasta mandarles a tomar por culo de la peor forma posible, pero han seguido llamando, a todas horas, hasta tres veces al día. Incluso he llamado al 900 809 400  (teléfono de jazztel para que no te llamen más) y aunque se ha reducido la frecuencia de las llamadas, siguen molestando.

He llegado a un punto en el que preferiría vivir sin Internet a contratar nada en esa maldita compañía.

He querido comentar esto en el blog para avisaros de que NO METAIS VUESTRO NÚMERO DE TELÉFONO EN LA WEB DE JAZZTEL. Empiezas haciendo una simple comprobación de cobertura, y acabas planteándote cambiar tu número de teléfono.

jueves, 14 de abril de 2011

Nokia C5-03: Configurar un PROXY para la conexión WIFI

Imagino que si estás en esta página es porque estás intentando salir a Internet a través del interfaz WIFI. Para que funcione necesitas definir un servidor PROXY y/o una dirección IP fija, y te has vuelto loco buscando dónde puedes especificar los parámetros de configuración de red en el maldito teléfono de los *******.

Bien, pues una vez sabes cómo hacerlo, no es difícil:
  1. En el menú principal (botón blanco del centro), selecciona Ajustes , Conectividad , Destinos de red y Añadir nuevo punto de acceso
  2. Ver puntos de acceso disponibles -SI
  3. Selecciona tu red WIFI (p.ej. MIWIFI), y si es necesario escribe la contraseña.
  4. Selecciona el destino : Internet
Se añadirá un nuevo punto de acceso a Internet , con el nombre de la red MIWIFI. No salgas todavía de esa pantalla.

Pincha (bueno, pon el dedazo) sobre el nombre de la red. Aparecerán las características básicas. Ve a opciones, y selecciona "Ajustes avanzados" ¡Ajá ! Ahí puedes meter la dirección de tu servidor proxy y/o tu IP fija.

Saludos.

martes, 22 de marzo de 2011

F1: 20 pasos para configurar un coche de carreras

Bueno, antes que nada aclarar que estas imágenes no son mías. Las hizo un amigo, Yago, pero me pidió que las subiera al blog para que las pudiera ver más gente.

Se trata de dos hojas con una lista de parámetros de configuración de un coche de carreras. Él las usa como referencia en simuladores de Fórmula 1.

Para ver (e imprimir) las fichas a tamaño completo, pulsa sobre las imágenes.

20 pasos para configurar un coche de carreras - Hoja 1 -

20 pasos para configurar un coche de carreras - Hoja 2 -

viernes, 11 de febrero de 2011

C#: Acceso directo a ficheros de Word 2010

- Introducción.

A veces es útil acceder directamente a un fichero de Word en formato .docx desde un programa en C#. Por ejemplo, yo lo uso para generar documentos personalizados más allá de lo que permite la opción "combinar correspondencia", insertar imágenes, etc.

Antes, los ficheros de word eran binarios y esas tareas había que hacerlas a través de la aplicación Word con  objetos COM. Desde la introducción del formato open XML (.docx), el mundo es un lugar mejor para los programadores.

- El meollo de la cuestión.

Un fichero .docx no es más que un fichero comprimido .zip , que contiene todos los ficheros e imágenes del documento de Word en formato XML.

Si quieres hacer la prueba, crea un fichero prueba.docx, escribe algo de texto, inserta una imagen, guárdalo, sal de Word, renómbralo como prueba.xml, y ábrelo. Verás un fichero XML, y varias carpetas. Entra en la carpeta "word", y abre el fichero "document.xml" Ahí está tu texto. Si abres la carpeta "media", verás tus imágenes.

Finalmente, cierra los ficheros, y renombra la carpeta prueba.zip por prueba.docx Podrás abrirlo sin problema con word.

En los siguientes apartados veremos cómo modificar el fichero a través de C# Voy a usar las siguientes variables:

string directorio_trabajo = Path.GetTempPath();
string fichero_entrada ="prueba";
string fichero_salida = "prueba2.docx";

... donde, evidentemente, directorio_trabajo es un directorio temporal donde comprimir/descomprimir los ficheros, fichero_entrada es el fichero "plantilla" que nos va a servir de base para generar la documentación, y fichero_salida va a contener el resultado.

En los ejemplos de código siguientes, vamos a suponer que fichero_entrada se encuentra dentro del directorio_trabajo.

- Descomprimiendo el fichero.

Hace años, cuando investigué la forma de acceder a ficheros OpenXML, C# no tenía forma de acceder a una carpeta ZIP. La solución que encontré (no muy elegante), es usar un programa externo para comprimir y descomprimir los ficheros. Busca (en Google) una versión libre de zip.exe y unzip.exe como esta....

http://stahlworks.com/dev/index.php?tool=zipunzip

Una vez tengas esos ejecutables, añádelos a tu proyecto, y usa este código para llamarlo:

string orden = "\"" + Application.StartupPath + "\\unzip.exe\"";
string parametros = "\"" + Application.StartupPath + "\\prueba.docx\"";


System.Diagnostics.Process p = new System.Diagnostics.Process();
System.Diagnostics.ProcessStartInfo info = new System.Diagnostics.ProcessStartInfo(orden);
info.UseShellExecute = true;
info.Arguments = parametros;
info.WorkingDirectory = dir_trabajo;
info.WindowStyle = System.Diagnostics.ProcessWindowStyle.Hidden;


p.StartInfo = info;
while (!p.Start()) ;
while (!p.HasExited) ;


Dependiendo de dónde esté tu fichero docx, tendrás que ajustar un poco las rutas de los ficheros en tu código.

- Cambiando el texto del fichero.

Dentro del árbol de ficheros que aparece tras descomprimir el .docx, debes abrir el fichero \word\document.xml.

La forma elegante de hacer los cambios es cargar el fichero XML con la clase XmlDocument , buscar los nodos que contienen el texto que quieres sustituir y volver a crear el fichero. Desgraciadamente, me pagan por programar rápido y no por programar elegantemente :)

La forma más rápida de hacer cambios es cargar el fichero XML en una cadena de texto:

// Leemos toda la cadena en un fichero.
string documento = File.ReadAllText(fichero + "\\word\\document.xml");


// Pon aquí los cambios que necesites. En el ejemplo, sustituyo la cadena "{nombre}" en 
// la plantilla por un nombre cualquiera.
documento = documento.Replace("{nombre}", "Armando Bronca Segura");


// Volvemos a grabar el fichero.
File.WriteAllText(fichero + "\\word\\document.xml", documento);

- Cambiando imágenes.

La forma más sencilla de cambiar imágenes en un documento es insertar una imagen desde Word en la plantilla, y luego sustituirla por otra en tu programa.

Las imágenes están en la carpeta \word\media\image1.gif , \word\media\image2.gif ... Antes de de escribir el programa, debes ver qué nombre le ha dado Word a tu imagen, para poder sustituirla por otra.

Si tienes un fichero con la nueva imagen, solo tienes que copiarlo usando la librería IO:

File.Copy ("mi_imagen.gif", @" \word\media\image2.gif");

Si quieres copiar el contenido del portapapeles como una imagen, usa este código:

if (Clipboard.GetDataObject() != null)
{
         IDataObject data = Clipboard.GetDataObject();


         if (data.GetDataPresent(DataFormats.Bitmap))
         {
                  Image image = (Image)data.GetData(DataFormats.Bitmap, true);
                  image.Save(Directory.GetCurrentDirectory() + "\\" + fichero + "\\word\\media\\image1.gif", System.Drawing.Imaging.ImageFormat.Gif);
         }
}


- Comprimiendo el fichero.

Una vez hechos todos los cambios, hay que volver a crear el documento .docx

string orden = "\"" + Application.StartupPath + "\\zip.exe\"";
string parametros = " -r -9 \"" + dir_salida + @"\prueba2.docx\" *";

System.Diagnostics.Process p = new System.Diagnostics.Process();
System.Diagnostics.ProcessStartInfo info = new System.Diagnostics.ProcessStartInfo(orden);
info.UseShellExecute = true;
info.Arguments = parametros;
info.WorkingDirectory = dir_trabajo;
info.WindowStyle = System.Diagnostics.ProcessWindowStyle.Hidden;

p.StartInfo = info;
while (!p.Start()) ;
while (!p.HasExited) ;

- Abriendo el fichero.


Probablemente querrás abrir el fichero con Word  desde tu aplicación para que el usuario no tenga que buscarlo y abrirlo él.

System.Diagnostics.Process p = new System.Diagnostics.Process();
System.Diagnostics.ProcessStartInfo info = new System.Diagnostics.ProcessStartInfo(plantilla);
info.UseShellExecute = true;
p.StartInfo = info;
while(!p.Start());

Saludos :)

miércoles, 9 de febrero de 2011

C#: Tareas multihilo en dos patás.

- Introducción.

Como probablemente ya sabes, hoy en día es importante construir tus aplicaciones de forma que aprovechen los múltiples "cores" de los procesadores y que sean capaces de seguir respondiendo al usuario aunque "por detrás" estén ejecutando procesos pesados.

En los viejos tiempos (más o menos, a principios del año pasado) , los programadores de C# usábamos el objeto BackgroundWorker para construir aplicaciones multihilo. Con la llegada de .NET 4, tenemos una nueva herramienta: los objetos Task.

A continuación encontrarás algunos fragmentos de código útiles para crear tareas y organizar su ejecución. Dejo de lado el tratamiento de excepciones y el paso de parámetros para otro artículo.

- Para crear una tarea.

Imagínate que tienes un formulario, y que quieres ejecutar una tarea cuando se carga (por ejemplo, leer unos datos de la base de datos).

using System.Threading.Tasks;
....
private void Callejero_Load(object sender, EventArgs e)


{
     Task t1 = new Task(
     () =>
           {
                  // Cargamos la lista de provincias en el control.
           });


      t1.Start();
}

Este  método, define una tarea, solicita que se empiece a ejecutar, y sale sin esperar a que termine. El formulario seguirá respondiendo al usuario mientras ejecuta otras tareas en segundo plano.

Una cosa importante es que no puedes acceder a los controles del formulario desde un hilo distinto al que lo ha creado. Por ejemplo, la tarea no podrá cambiar el valor de una caja de texto del formulario directamente. Más adelante veremos un ejemplo de cómo hacerlo.


- Definir una tarea que se ejecute cuando otra termine.

Si tenemos una tarea T1 (como la anterior), y queremos definir una tarea T2 que se ejecute a continuación...


Task t2 = t1.ContinueWith(
ant =>
{
           // Hacer más cosas
});


No es necesario llamar a Start() en una tarea de continuación (de hecho da un error). La tarea se arrancará automáticamente cuando T1 termine.

- Definir una tarea que se ejecute cuando varias tareas terminen.

Si tenemos dos tareas T1 y T2 que se van a ejecutar en paralelo, y queremos definir una tarea T3 que arranque cuando las otras dos terminen ...


Task t3 = Task.Factory.ContinueWhenAll(
       new[] { t1, t2 },
        tasks =>
              {
                   // Hacemos más cosas
              }
 );

- Definir una tarea que se ejecute cuando otra termina, y actualice controles de nuestro formulario WPF.

Este es un caso muy típico. Imagínate que tienes un formulario que, cuando pulsas un botón, arranca una tarea T1. Cuando la tarea empieza, apagas un botón para que el usuario no le vuelva a dar. Cuando la tarea termina, quieres volver a habilitarlo. Nada más fácil.


Task t_final = t_1.ContinueWith(
        ant =>
       {
              btn_procesar.IsEnabled = true;
       }, TaskScheduler.FromCurrentSynchronizationContext());




- Definir una tarea que se ejecute cuando terminan otras, y actualice controles de nuestro formulario.



Task tarea_final = Task.Factory.ContinueWhenAll(
                        new[] { task_eventos, task_estados, task_info, task_parametros, task_control, task_canont },
                        tasks =>
                        {
                            // Accedemos a controles de nuestro formulario.
                        }, 
                        CancellationToken.None, 
                        TaskContinuationOptions.None,
                        TaskScheduler.FromCurrentSynchronizationContext()
                    );

viernes, 21 de enero de 2011

Debian Squeeze: habilitar sesiones X remotas.

Por motivos de seguridad, por defecto en las distribuciones de Debian el servidor de X Windows no escucha en ningún puerto de red.

En versiones anteriores, lo único que había que hacer era irse al fichero /etc/X11/xinit/xserverrc y cambiar la línea:
 exec /usr/bin/X -nolisten tcp "$@"
por
exec /usr/bin/X "$@"
Desgraciadamente, si usas GNOME ahora eso no funciona. El problema está en que el programa GDM3 siempre añade la opción "nolisten" a menos que se configure específicamente para que no lo haga. Hay que ir al fichero  /etc/gdm3/daemon.conf , y done pone ...
[security]

... añadir ...
[security]
DisallowTCP=false
Reinicia las X (o el equipo), y deberías ver al servidor escuchando en el puerto 6000:
~$ nmap -sT 192.168.0.20


Starting Nmap 5.00 ( http://nmap.org ) at 2011-01-21 10:09 CET
Interesting ports on 192.168.0.20:
Not shown: 989 closed ports
PORT     STATE SERVICE
...
6000/tcp open  X11
6001/tcp open  X11:1
...
Nmap done: 1 IP address (1 host up) scanned in 0.05 seconds
Saludos.