Foros Club Delphi

Foros Club Delphi (https://www.clubdelphi.com/foros/index.php)
-   FireMonkey (https://www.clubdelphi.com/foros/forumdisplay.php?f=50)
-   -   StringGrid Concatenar Celdas (https://www.clubdelphi.com/foros/showthread.php?t=88044)

Lenny 07-04-2015 18:04:32

StringGrid Concatenar Celdas
 
Estimados, tengo una pequeño problema en un proyecto en el cual debo crear un frontend para escritorio de una sistema web en el cual uno de los principales requisitos es no modificar la estructura de la BD.
Hasta el momento todo ok pero me tope con el siguiente problema:

Las BD tienen los nombres de los usuarios/clientes, apellido paterno y materno guardados en distintos campos y en la aplicación web cuando se muestran los usuarios aparecen en la columna "nombres" (que solo tiene el o los nombres en la BD) con nombre y apellidos concatenados, necesitan que se vea igual en la versión de escritorio.

Generalmente utilizo este código en el evento StringGrid1DrawColumnCell para hacer algo similar en el mismo sistema (funciona perfecto):

Código Delphi [-]
begin
  if (Sender as TStringGrid).Cells[3, Row] = '0' then
  begin
    (Sender as TStringGrid).Cells[3, Row] := 'No';
    end
    else
  if (Sender as TStringGrid).Cells[3, Row] = '1' then
  begin
    (Sender as TStringGrid).Cells[3, Row] := 'Si';
end;

He tratado de cambiar el código para que me permita poder cambiar la celda nombre por la concatenación de Nombre, apellido Paterno y Materno y no logro hacerlo, de antemano agradezco cualquier ayuda, quedo atento a sus comentarios, saludos.

Utilizo XE7, MySQL, Firemonkey, Query y StringGrid.

ecfisa 07-04-2015 20:14:42

Hola Lenny.

La verdad es que no sé si terminé de comprender claramente la situación... De lo que entendí, no veo que relación pueda tener el evento OnDrawColumnCell con lo que buscas hacer.
Si deseas mostrar la unión de nombre-apellido en una columna, podes hacerlo al momento de cargar el TStringGrid desde la tabla de tu bd.

Un ejemplo de lo que te digo con la tabla CLIENTS que trae Delphi de facto:
Código Delphi [-]
procedure TForm1.FormCreate(Sender: TObject);
var
  SG: TStringGrid;
begin
  SG := StringGrid1;
  SG.FixedRows := 1;
  SG.RowCount  := 2;
  SG.ColCount  := 3;
  SG.ColWidths[2] := 200;
  SG.Rows[0].CommaText := 'FirstName,LastName,FullName';
  with Query1 do
  begin
    Close;
    SQL.Clear;
    SQL.Add('SELECT FIRST_NAME, LAST_NAME,');
    SQL.Add('FIRST_NAME+'' ''+LAST_NAME AS FULL_NAME FROM CLIENTS');
    Open;
    while not Eof do
    begin
      SG.Cells[0,SG.RowCount-1] := FieldByName('FIRST_NAME').AsString;
      SG.Cells[1,SG.RowCount-1] := FieldByName('LAST_NAME').AsString;
      SG.Cells[2,SG.RowCount-1] := FieldByName('FULL_NAME').AsString;
      SG.RowCount:= SG.RowCount + 1;
      Next;
    end;
  end;
end;
Si deseas realizar la concatenación desde Delphi, reemplaza estas líneas:
Código Delphi [-]
...
    SQL.Text := 'SELECT FIRST_NAME, LAST_NAME FROM CLIENTS';
...

    SG.Cells[2,SG.RowCount-1] := FieldByName('FIRST_NAME').AsString + ' ' +
                                 FieldByName('LAST_NAME').AsString;
...

Resultado del ejemplo:



Saludos :)

AgustinOrtu 07-04-2015 20:56:36

O si usas algun componente Query para acceder a los datos, podes traerte el nombre completo de la BD

Ejemplo que yo uso en Firebird:

Código SQL [-]
SELECT P.*, (P.Apellido || '', '' || P.Nombre) AS NombreCompleto FROM Persons P

Por supuesto despues debes usar el "nuevo campo" 'NombreCompleto'

Código Delphi [-]
  ShowMessage(Query.FieldByName('NombreCompleto').AsString);

Deberias buscar como concatenar strings en MySQL, cada motor podria tener distinta sintaxis

Saludos

Lenny 08-04-2015 09:43:02

Estimados, ninguno de los 2 ejemplos me sirvió, el de ecfisa con error y es por que estoy trabajando con Firemonkey, el de AgutinOrtu en teoría debería funcionar, lo edite según mis necesidades pero me lanza error preguntando por el primer registro "Field 'IDUSUARIO' not found" siendo que en ningún lado hago referencia a el.

Me explico mejor, necesito de una tabla llamada "usuario" concatenar las columnas "nombres", apellido "paterno" y "materno", todo esto en un StrinGrid, por que? por que el cliente tiene una sistema web (JAVA) y quiere un frontend para escritorio y quiere que al menos las ventanas que mas usan (en este caso usuarios) se vean como en la web, donde la columna "nombres" muestra el nombre completo del usuario.
El problema principal es que uno de los requisitos mas importantes es trabajar con la base de datos como esta, sin agregar, cambiar menos quitar algo de la BD lo cual dificulta un poco el proyecto.
¿Por que en el evento OnDrawColumnCell? En algún momento (como ahora) me toco cambiar los colores de ciertas celdas que cumplan X condición, con el tiempo necesite hacer lo mismo pero en ves de cambiar el color cambiar los textos si cumplían cierta condición (True / False), por ejemplo un checkbox que en ves de aparecer 0 / 1 aparezca Activado / Desactivado en el StrinGrid. Como modifique el mismo código que estaba en OnDrawColumnCell y me funciono lo seguí usando de esa manera (hasta ahora) y nunca he tenido problemas.
¿Por que StrinGrid y la petición? El cliente quiere ver todos los usuarios como en la web (en una lista) y Firemonkey solo tiene Grid y StringGrid.

Espero haber aclarado un poco mas mi problema, si necesitan mas detalles o explicar algo mas estaré pendiente, de antemano muchas gracias.

AgustinOrtu 08-04-2015 09:48:41

El codigo de ecfisa te genera un error? Cual es?

Si te sale en algun lado que no encuentra un campo es porque "no se lo estas dando"

Estas usando LiveBindings?

Lenny 08-04-2015 09:52:39

Como comente antes utilizo Firemonkey, en VCL funciona perfecto por eso el error, me a pasado con varios códigos que tuve que rehacer pero en este caso simplemente el componente StringGrid no soporta ciertas características que vienen en el código, saludos y gracias.

PD: si, estoy usando LiveBindings (Firemonkey).

Lenny 08-04-2015 10:15:33

Solucionado
 
estimados, muchísimas gracias por la ayuda, a pura prueba y error y con lo dicho ya lo solucione:

Código Delphi [-]
procedure TUsuarios.StringGrid1DrawColumnCell(Sender: TObject;
  const Canvas: TCanvas; const Column: TColumn; const Bounds: TRectF;
  const Row: Integer; const Value: TValue; const State: TGridDrawStates);
begin
      if (Sender as TStringGrid).Cells[3, Row] = 'Fals' then
  begin
    (Sender as TStringGrid).Cells[3, Row] := 'No';
    end
    else
  if (Sender as TStringGrid).Cells[3, Row] = 'True' then
  begin
    (Sender as TStringGrid).Cells[3, Row] := 'Si';
  end;
(Sender as TStringGrid).Cells[2, Row] := (Sender as TStringGrid).Cells[1, Row] + ' ' + (Sender as TStringGrid).Cells[6, Row];
end;

los primeros 2 IF cambian en el StringGrid los valores True / Fals por Si / NO (puede ser cualquier otro valor, Activado / Desactivado, etc.) y la ultima linea hace lo que necesito, cargar en el StrinGrid en la columna "Nombres" la concatenación del nombres y apellido del usuario cumpliendo con los requisitos de no tocar la estructura de la base de datos.

Solo me queda saber si existe una forma mas elegante y no tan artesanal de hacer esto, y donde mas aparte de DrawColumnCell puedo colocar el código, ya que no me sirve en el Form.Create por si se actualizan los datos.

Gracias por la ayuda y quedo atento a sus comentarios!.

AgustinOrtu 08-04-2015 11:02:37

Creo que podes asignar dinámicamente a tu fuente de datos, en el campo Nombre, un manejador en el evento OnGetText

Por ejemplo defino esto en la sección private del form

Código Delphi [-]
procedure Grid_Turnos_Dia_DuracionOnGetText(Sender: TField; var Text: string; DisplayText: boolean);

La asignación dinámica

Código Delphi [-]
Dataset.FieldByName('Duracion').OnGetText := Grid_Turnos_Dia_DuracionOnGetText;

Y la implementación

Código Delphi [-]
procedure TTFMain.Grid_Turnos_Dia_DuracionOnGetText(Sender: TField; var Text: string; DisplayText: boolean);
begin
    if not(Sender.IsNull) then
    begin
        Sender.Alignment := taRightJustify;
        Text := FormatDateTime('HH:MM', Sender.Value);
    end;
end;

Luego con LiveBindigs te recomiendo revisar aca


El evento que usaste, OnDrawCell yo lo dejaría para manejar el aspecto visual del grid en si, es decir, fuente, color de fondo, etc

ecfisa 08-04-2015 11:53:59

Hola Lenny.

Debo reconocer que no leí (o se me pasó) "Firemonkey" al leer el mensaje :o
Cita:

Empezado por Lenny (Mensaje 490909)
...
Utilizo XE7, MySQL, Firemonkey, Query y StringGrid.

Muevo el tema a dicho foro.

Saludos :)


La franja horaria es GMT +2. Ahora son las 19:55:00.

Powered by vBulletin® Version 3.6.8
Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
Traducción al castellano por el equipo de moderadores del Club Delphi