PDA

Ver la Versión Completa : RxDBGrid y los CheckBox


Nuria
31-03-2004, 12:46:47
Hola!

Mirándo el ejemplo del RxDBGrid que tiene en su página web el compi (gracias...;) cadetill (http://www.clubdelphi.com/~cadetill/)), he creado según mis necesidades el mío propio, pero tengo el siguiente problemilla con los CheckBox.

Trabajo con Delphi 7 y con IBX.

Tengo el RxDBGrid enlazado a un TIBQuery, que yo le he añadido un campo calculado SEL de tipo boolean que lo voy a usar para marcar o desmarcar las líneas del grid. Esto es lo que he hecho:

procedure RxDBGrid1CellClick(Column: TColumn);
begin
if RxDBGrid1.SelectedIndex = 3 then begin // Columna donde tengo el checkbox
if IBQ.FieldByName('SEL').AsBoolean then begin
IBQ.FieldByName('SEL').AsBoolean := False;
end
else begin
IBQ.FieldByName('SEL').AsBoolean := True;
end
end;
end;

Cuando hago click en la casilla del CheckBox me la marca o me la desmarca según corresponda, hasta ahí bien. Pero vuelvo a hacer un click en la misma casilla y me desaparece el check y me aparece la palabra True/False. Si voy 'clickeando' en casillas distintas funciona a la perfección pero si es en la misma no.
:confused: Qué podría hacer para que no me aparecieran las palabras True/False? Es mejor hacerlo de otra manera?

Muchas gracias....

Un Saludillo de esta humilde 'Piltrafilla'.

Nuria
31-03-2004, 14:01:13
Sólo añadir una cosilla más a lo dicho anteriormente, por si tuviera algo que ver. Este es el código que tengo en el evento OnDrawColumnCell del RxDBGrid.


procedure TfFacturas.RxDBGrid1DrawColumnCell(Sender: TObject;
const Rect: TRect; DataCol: Integer; Column: TColumn;
State: TGridDrawState);
const CtrlState : array[Boolean] of Integer = (DFCS_BUTTONCHECK,
DFCS_BUTTONCHECK or DFCS_CHECKED);
var CheckBoxRectangle : TRect;
begin

// Pintamos toda la fila de la celda booleana
// Para hacerlo tenemos que hacer referencia al campo booleano
if IBQ.AsBoolean then
begin
RxDBGrid1.Canvas.Brush.Color := clInfobk;
RxDBGrid1.Canvas.Font.Style := [fsBold];
RxDBGrid1.Canvas.Font.Color := ClBlack;
RxDBGrid1.DefaultDrawColumnCell(Rect,Datacol,Column,State);
end;

// campo Check
if Column.Field.DataType = ftBoolean then
begin
// Pintamos solo la celda booleana
if Column.Field.AsBoolean then
begin
RxDBGrid1.Canvas.Brush.Color := clInfobk;
RxDBGrid1.Canvas.Font.Style := [fsBold];
RxDBGrid1.Canvas.Font.Color := ClBlack;
RxDBGrid1.DefaultDrawColumnCell(Rect,Datacol,Column,State);
end;

RxDBGrid1.Canvas.FillRect(Rect);
CheckBoxRectangle.Left := Rect.Left + 2;
CheckBoxRectangle.Right := Rect.Right - 2;
CheckBoxRectangle.Top := Rect.Top + 2;
CheckBoxRectangle.Bottom := Rect.Bottom - 2;
DrawFrameControl(RxDBGrid1.Canvas.Handle, CheckBoxRectangle,
DFC_BUTTON, CtrlState[Column.Field.AsBoolean]);
end;
end;

mmmm.... :rolleyes: , de donde he podido sacar este código... :p

Saludillos!

__cadetill
31-03-2004, 16:20:49
Hola piltrafilla :D

A ver, algunos comentarios

En tu código veo....

if IBQ.AsBoolean then

Esto te compila?? :eek: IBQ no es el TIBQuery?? Te falta el campo!!! :D

Otra cosilla

Si la celda booleana con valor true la pintas de la misma manera que toda la fila de la cual el campo booleano está en true... te sobra un DefaultDrawColumnCell. Te tendría que quedar así:


procedure TfFacturas.RxDBGrid1DrawColumnCell(Sender: TObject;
const Rect: TRect; DataCol: Integer; Column: TColumn;
State: TGridDrawState);
const CtrlState : array of Integer = (DFCS_BUTTONCHECK,
DFCS_BUTTONCHECK or DFCS_CHECKED);
var CheckBoxRectangle : TRect;
begin

// Pintamos toda la fila de la celda booleana
// Para hacerlo tenemos que hacer referencia al campo booleano
if IBQ[B]CampoBooleano.AsBoolean then
begin
RxDBGrid1.Canvas.Brush.Color := clInfobk;
RxDBGrid1.Canvas.Font.Style := [fsBold];
RxDBGrid1.Canvas.Font.Color := ClBlack;
RxDBGrid1.DefaultDrawColumnCell(Rect,Datacol,Column,State);
end;

// campo Check
if Column.Field.DataType = ftBoolean then
begin
RxDBGrid1.Canvas.FillRect(Rect);
CheckBoxRectangle.Left := Rect.Left + 2;
CheckBoxRectangle.Right := Rect.Right - 2;
CheckBoxRectangle.Top := Rect.Top + 2;
CheckBoxRectangle.Bottom := Rect.Bottom - 2;
DrawFrameControl(RxDBGrid1.Canvas.Handle, CheckBoxRectangle,
DFC_BUTTON, CtrlState[Column.Field.AsBoolean]);
end;
end;


Mas cosillas

He probado tu código (sólo que atacando a Paradox ya que he usado la demo de la web) y, a mi no me hace el efecto que indicas :(
Los cambios que he hecho son....


procedure TDbGrid.RxDBGrid1CellClick(Column: TColumn);
begin
RxDBGrid1.DataSource.DataSet.Edit;
if RxDBGrid1.SelectedIndex = 0 then // Columna donde tengo el checkbox
RxDBGrid1.DataSource.DataSet.FieldByName('Preferred').AsBoolean :=
not RxDBGrid1.DataSource.DataSet.FieldByName('Preferred').AsBoolean;
RxDBGrid1.DataSource.DataSet.Post;
end;


Si puedes, hazlos tambien en la demo, a ver si te pasa (ah, por cierto, si lo haces, recuerda en poner la pripiedad RequestLive en true del TQuery ;))

Nuria
31-03-2004, 17:04:42
Hola xiquitín!


Esto te compila?? IBQ no es el TIBQuery?? Te falta el campo!!!


Cierto, es IBQSEL, me equivoqué al escribir... hoy estoy un poco 'empana'.

He probado lo que me has dicho en la demo y funciona a la perfección. He intentado hacerlo así:

RxDBGrid1.DataSource.DataSet.Edit;
if RxDBGrid1.SelectedIndex = 0 then // Columna donde tengo el checkbox
RxDBGrid1.DataSource.DataSet.FieldByName('SEL').AsBoolean :=
not RxDBGrid1.DataSource.DataSet.FieldByName('SEL).AsBoolean;
RxDBGrid1.DataSource.DataSet.Post;


Pero me da un error porque uso un TIBQuery:

...DatabaseError.... 'IBQ: Cannot modify a read-only dataset'

Los TIBQuery no tienen la propiedad RequestLive y que yo sepa nada parecido. Por lo tanto me sigue sin funcionar... :( .

Voy a seguir investigando y probando, cuando lo solucione que lo solucionaré :rolleyes: te cuento. Muchas gracias 'apañero'.

__cadetill
31-03-2004, 17:15:02
Los TIBQuery no tienen la propiedad RequestLive y que yo sepa nada parecido. Por lo tanto me sigue sin funcionar... :( .

En este caso tienes 2 opciones, o bien usas un TIBDataset o bien la unión TIBQuery + TIBUpdateSQL

Te lo dejo a tu elección ;)

Nuria
31-03-2004, 17:39:47
En este caso tienes 2 opciones, o bien usas un TIBDataset o bien la unión TIBQuery + TIBUpdateSQL

:eek: Pues fíjate tú, que nunca he usado el TIBUpdateSQL, acabo de meterme hace muy poco con los IBX, he estado investigando y creo que lo voy a usar en bastantes ocasiones...Gracias.

No creo que pueda hacer lo de los Check con la modificación que me dijiste antes porque el campo Boolean NO es un campo de la tabla propiamente dicho, sino un campo calculado que me he creado yo. Ya veré como lo hago finalmente.

Un saludillo! ;)

Nuria
31-03-2004, 19:06:46
He encontrado una solución, no me gusta mucho pero de esta manera ya no me desaparece el check cuando marco dos veces sobre la casilla.

procedure RxDBGrid1CellClick(Column: TColumn);
begin
if RxDBGrid1.SelectedIndex = 3 then begin // Donde tengo el check
RxDBGrid1.DataSource.DataSet.FieldByName('SEL').AsBoolean :=
not RxDBGrid1.DataSource.DataSet.FieldByName('SEL').AsBoolean;
RxDBGrid1.SelectedIndex := 1; // Paso el foco a otra celda.
end;
end;


Sigo buscando otra solución mejor.

Saludos! ;)

__cadetill
31-03-2004, 19:30:57
No creo que pueda hacer lo de los Check con la modificación que me dijiste antes porque el campo Boolean NO es un campo de la tabla propiamente dicho, sino un campo calculado que me he creado yo. Ya veré como lo hago finalmente.
Bueno, supongo que será un entero con valores 1 o 0, no?
Si es así, prueba a poner....


if Column.Field.DataType = ftInteger then
begin
// Pintamos solo la celda booleana
if Column.Field.AsBoolean and (Pintar.ItemIndex = 1) then
begin
RxDBGrid1.Canvas.Brush.Color := clInfobk;
RxDBGrid1.Canvas.Font.Style := [fsBold];
RxDBGrid1.Canvas.Font.Color := ClBlack;
RxDBGrid1.DefaultDrawColumnCell(Rect,Datacol,Column,State);
end;

RxDBGrid1.Canvas.FillRect(Rect);
CheckBoxRectangle.Left := Rect.Left + 2;
CheckBoxRectangle.Right := Rect.Right - 2;
CheckBoxRectangle.Top := Rect.Top + 2;
CheckBoxRectangle.Bottom := Rect.Bottom - 2;
DrawFrameControl(RxDBGrid1.Canvas.Handle, CheckBoxRectangle,
DFC_BUTTON, CtrlState[Column.Field.AsInteger = 1]);
end;


En negrita las lineas que cambian. De esta manera te evitas el campo calculado. Luego, en el OnClick, sólo has de actualizar susodicho campo.

Atención: esta manera tiene un problema, y es que afectará a todos los campos de tipo Integer.
Solución: en lugar de mirar el tipo de campo, mirar por el nombre de campo
ej: if Column.Field.FieldName = 'MiCampoInteger' then

Espero te sirva

Nuria
01-04-2004, 10:20:55
Muchas gracias de nuevo.... ;)

He probado lo que tú me comentabas y me hace lo mismo pero en vez de ponerme true/false pone 0/1... :confused: .

Como te dije antes medianamente lo conseguí arreglar, pasando el foco a otra celda. Pero ahora además, me hace otra cosa ... cuando el grid tiene un montón de líneas pues me aparece el scroll vertical como es lógico, marco varios check, pincho en el scroll y voy bajando líneas, cuando vuelvo arriba me ha desmarcado todos los checks :( . En tú demo lo he probado y funciona correctamente.

Así que sigo dandole vueltas al asunto...

Un saludito!

__cadetill
01-04-2004, 12:28:23
Hola niña

Bueno, he hecho una prueba y ya he visto lo que te pasa :D
En la prueba me pasaba lo mismo que a ti y decidí mirar qué diferencias había entre la configuración de un Grid y el otro

Pues bien, el de la demo está en Option.dgEditing = false, es decir, no editable

Esto puedes controlarlo, si quieres, en el evento OnColEnter, mirar si es la columna del campo Check y si lo es, poner dgEditing a false, sino, ponerlo a true

¿qué te parece la idea?

Nuria
01-04-2004, 13:06:03
Hola !


Option.dgEditing = false,....Esto puedes controlarlo en el evento OnColEnter


Lo he probado y sigue haciéndome las mismas cosas raras :( ... se me resiste.

Lo que me esta dejando loca es lo de la barra del scroll, marco algunos checks y cuando pincho sobre las flechitas del scroll, van desplazándose las líneas y cuando alguna de esas líneas ya no la veo por pantalla y vuelvo a mostrarla me la muestra desmarca. Y si pincho directamente dentro del scroll me las desmarca todas... :mad: .

Sigo peleandome con el RxGrid....

Mil gracias! ;)