Foros Club Delphi

Foros Club Delphi (https://www.clubdelphi.com/foros/index.php)
-   Varios (https://www.clubdelphi.com/foros/forumdisplay.php?f=11)
-   -   cual es la mejor forma de llamar a un formulario de modificación de datos (https://www.clubdelphi.com/foros/showthread.php?t=75182)

novato_erick 03-08-2011 17:58:56

cual es la mejor forma de llamar a un formulario de modificación de datos
 
Hola nuevamente hace día he estado trabajando con formularios el cual se me han presentado inconvenientes, me he apoyado en literatura por ejemplo la cara ocultad e delphi, delphi7, delphi handbook (para uso de ribbon). Algunos de ellos brindan técnicas la cual realizo pero una cosa es la teoría y la otra es la practica ustedes deben de saber eso.

Gracias a este foro he encontrado personas que me han ayudado no dándome todo fácil pero si guiándome lo suficiente y a ellos les agradezco.

Siguiendo el asunto de formularios el inconveniente que poseo actualmente es que necesito utilizar mi formulario de ingreso de registros también como formulario de modificación de registro mi código actual algunos los conocen es.


Código Delphi [-]
procedure MuestraVentana(ventana: string; modo: string);
begin
 if ventana = 'Clientes' then
  begin
    if FrmCliente = nil then
    begin
      FrmCliente := TfrmCliente.create(FrmCliente);
      Try
      FrmCliente.Parent := FrmPrincipal.Panel4;
      FrmCliente.Caption := 'Clientes';
      dmacceso.cdsClientes.Active := True;
      dmacceso.cdsClientes.Last;
      dmacceso.cdsClientes.Open;
      dmacceso.cdsClientes.Insert;
      FrmCliente.Show;
      finally
         If FrmCliente.Caption <> ' ' then
         Begin
           FrmPrincipal.TabSet1.Tabs.add(FrmCliente.Caption);
           FrmPrincipal.TabSet1.TabIndex := FrmPrincipal.TabSet1.Tabs.Count - 1;
         end;
   end;
end;

    if modo = 'Editar' then
        begin
            frmCliente.btnModificar.Visible := True;
            frmCliente.btnGuardar.Visible := False;
            dmacceso.cdsClientes.Edit;
            frmCliente.Show;
    end;

  end;

Este es el boton de llamar al formulario para ser modificado

Código Delphi [-]
procedure TfrmFicheroCliente.aModificarExecute(Sender: TObject);
begin
  pais := dmAcceso.cdsClientesPais.text;
  dmacceso.cdsClientes.Edit;
  MostrarRegistroYGuardar;
end;

Este es un procedimiento de modificar en el boton

Código Delphi [-]
procedure TfrmCliente.btnModificarClick(Sender: TObject);
begin
  dmacceso.cdsClientesPAIS.Value := cbexPais.Text;
  dmacceso.cdsclientes.edit;
  dmacceso.cdsClientes.ApplyUpdates(0);

  if Application.MessageBox('Cliente Modificado',
  'Atención',(MB_OK + MB_ICONINFORMATION))= mb_ok then
  begin
    frmcliente.Close;   //            Estas dos lineas de código en particular no 
    frmFicherocliente.Show;//     me hacen nada cuando doy aceptar. Alguna idea como mejorar eso?
  end

end;

Como dije se me están presentando inconvenientes por ejemplo Al estar el formuolario abierto si no le doy clic algun dbEdit no se muestran los datos que quiero modificar..

Saludos;

novato_erick 03-08-2011 18:08:08

ah se me olvidaba este es mi procedimiento de MostrarRegistroyGuardar

Código Delphi [-]
procedure TfrmFicheroCliente.MostrarRegistroYGuardar;
begin
  utiles.MuestraVentana('Clientes', 'Editar'); //aqui llama al modo de Editar
end;

oscarac 03-08-2011 18:10:22

mmmm me parece algo complicado tu codigo.. tienes muchas cosas que son redundantes

por ejemplo
Código Delphi [-]
 
dmacceso.cdsClientes.Active := True;  // esto es igual que open
dmacceso.cdsClientes.Last; // no es necesatio cuando insertas pq se va al final
dmacceso.cdsClientes.Open;
dmacceso.cdsClientes.Insert;

yo particularmente hago esto

Boton Modificar del form donde muestro los datos
Código Delphi [-]
 
procedure TfrmCatalogoAnexo.btnModificaClick(Sender: TObject);
begin
  btn_opcion := 1;
  qryAnexos.Edit;
  frmCatalogoAnexod := TfrmCatalogoAnexod.Create(nil);
  frmCatalogoAnexod.Show
end;

y en el create del formulario invocado hago mas o menos esto
Código Delphi [-]
 
 
procedure TfrmCatalogoAnexod.FormCreate(Sender: TObject);
var i: Integer;
begin
  if frmCatalogoAnexo.btn_opcion =1 then
    begin
      edtCodigo.Enabled := False;
      ComboBox1.ItemIndex := StrToInt(frmCatalogoAnexo.qryAnexosTipoAnexo.Value) - 1;
    end;
  if frmCatalogoAnexo.btn_opcion >= 1 then
        edtTipoAnexoExit(sender);
  if frmCatalogoAnexo.btn_opcion = 2 then
    begin
      btnGrabar.Caption := 'Eliminar';
      for i := 0 to (Self.ComponentCount - 1) do  begin
        if (Self.Components[i] is TDBEdit) or (Self.Components[i] is TComboBox) then begin
          TEdit(Self.Components[i]).Enabled := False;
        end;
      end;
    end;
end;

y en el boton de grabar hago mas o menos esto

Código Delphi [-]
 
procedure TfrmCatalogoAnexod.btnGrabarClick(Sender: TObject);
begin
  if frmCatalogoAnexo.btn_opcion <> 2 then
    begin
      frmCatalogoAnexo.qryAnexosTipoAnexo.AsString := FormatFillStringNumber (IntToStr(ComboBox1.ItemIndex + 1),2);
      frmCatalogoAnexo.qryAnexosEmpresa.AsString := dmGlobal.g_CodigoEmpresa;
      frmCatalogoAnexo.qryAnexos.Post;
    end;
  if frmCatalogoAnexo.btn_opcion = 2 then
    frmCatalogoAnexo.qryAnexos.Delete;
  Close;
  frmCatalogoAnexo.dbgAnexo.SetFocus;
end;

novato_erick 03-08-2011 18:29:26

hola oscarac respecto a lo reduntante he notado que:

Código Delphi [-]
dmacceso.cdsClientes.open; // no me abre el DataSet al llamar el formulario

dmacceso.cdsClientes.Last; // si no lo utilizaba en mi formulario para registrar datos me eliminaba no se ponia de ultimo. y por lo general me eliminaba mi primer registro.

Por eso los utilizo... Se me comporta extraño el formulario cuando uso métodos que se consideran "apropiados".

Saludos;

oscarac 03-08-2011 18:45:01

ohh ya
pero eso depende de la instancia y lo que quieras hacer con el registro activo

Caral 03-08-2011 19:22:50

Hola
Coloca un checkbox en el form:
Código Delphi [-]
procedure TFrmPrincipal.CheckBox1Click(Sender: TObject);
begin
   If Checkbox1.Checked = True then
   begin
   Checkbox1.Caption:= 'Insertar';
   end
   else
   Checkbox1.Caption:= 'Modificar';
end;

procedure MuestraVentana(ventana: string; modo: string);
begin
 if ventana = 'Clientes' then
  begin
    if FrmCliente = nil then
    begin
      FrmCliente := TfrmCliente.create(FrmCliente);
      Try
      FrmCliente.Parent := FrmPrincipal.Panel4;
      FrmCliente.Caption := 'Clientes';
      dmacceso.cdsClientes.Active := True;

      If Checkbox1.Checked = True then
      begin
      // insertar
      dmacceso.cdsClientes.Last;
      dmacceso.cdsClientes.Open;
      dmacceso.cdsClientes.Insert;
      FrmCliente.Show;
      end
      else
      begin
      // modificar
      dmacceso.cdsClientes.Open;
      dmacceso.cdsClientes.Edit;
      FrmCliente.Show;
      end;

      finally
         If FrmCliente.Caption <> ' ' then
         Begin
           FrmPrincipal.TabSet1.Tabs.add(FrmCliente.Caption);
           FrmPrincipal.TabSet1.TabIndex := FrmPrincipal.TabSet1.Tabs.Count - 1;
         end;
   end;
end;

procedure TfrmCliente.btnModificarClick(Sender: TObject);
begin
  dmacceso.cdsClientesPAIS.Value := cbexPais.Text;
  dmacceso.cdsClientes.ApplyUpdates(0);

  if Application.MessageBox('Cliente Modificado',
  'Atención',(MB_OK + MB_ICONINFORMATION))= mb_ok then
  begin
    frmcliente.Close;   //            Estas dos lineas de código en particular no 
    frmFicherocliente.Show;//     me hacen nada cuando doy aceptar. Alguna idea como mejorar eso?
  end

end;

end;
saludos

Chris 03-08-2011 19:51:48

A cómo dice Oscarac, parece que te has complicado en algo que puede ser simple. No te preocupes, estás empezando y es lo más normal de todo el mundo. En mi caso, fíjate lo que hago:

Declaro dos contructores en la clase (los llamo "visores de registro", para diferenciarlos de los "exploradores de registro"). La declaración de los constructores es más o menos así:
Código Delphi [-]
type
    TClientesViewer =  class(TBaseViewer)
    published
        constructor ShowRecord(AOwner: TComponent; RecordID: Variant;
                               const ReadOnly: Boolean = False;
                               const UpdateTransaction: TpFIBTransaction = nil;
                               ...);
        constructor NewRecord(AOwner: TComponent; const RecordData: PString = nil);

Luego en la implementación de cada constructor preparo el enlace a datos de la forma más adecuada para el modo en el que se trabajará. El proceso de guardar los cambios o ya sea un nuevo registro, es casi lo mismo. En síntesis solo se requiere de una llamada a Post.

Saludos,
Chris

roman 03-08-2011 20:15:54

Desde luego, depende de como se hace el enlace, pero no me cuadra mucho lo de los dos constructores. A la ventana de edición de registro debería darle lo mismo si está editando un registro existente o uno nuevo. La diferencia debiera darla quien abre la ventana poniendo el dataset en modo de edición o de inserción.

// Saludos

Chris 03-08-2011 21:24:22

Cita:

Empezado por roman (Mensaje 408379)
La diferencia debiera darla quien abre la ventana poniendo el dataset en modo de edición o de inserción.

Terminarías repitiendo código Román. Tomé este diseño por la experiencia. Si quieres crear un nuevo registro, solo falta con:
Código Delphi [-]
TClientesViewer.NewRecord(Self);
Una simple línea de código.
Si lo que quieres modificar otro registro, también solo faltará con una simple línea de código
Código Delphi [-]
TClientesViewer.ShowRecord(Self, Table1.Fields['id'].AsString);

Cita:

Empezado por roman (Mensaje 408379)
A la ventana de edición de registro debería darle lo mismo si está editando un registro existente o uno nuevo.

En cierta forma así es. Prácticamente lo único que hacen los constructores es adaptar la interfaz de la ventana para ligeramente diferenciarse entre modificación o inserción de un registro.

Saludos,
Chris

novato_erick 04-08-2011 05:33:21

como dijo roman
Cita:

A la ventana de edición de registro debería darle lo mismo si está editando un registro existente o uno nuevo. La diferencia debiera darla quien abre la ventana poniendo el dataset en modo de edición o de inserción.
Es lo que trato de decirles: En teoría son argumentos que exponen los autores en sus reconocidos libros que "Funcionan" sin embargo en la programación me he percatado que es a base de experiencia lo que realmente manda. Agradezco muchísimo sus aportes caral, roman, chris,oscarac y quien me falta por mencionar la cual son muchos Gracias.

Lo que entiendo de chris es que voy a crear una clase que puedo mas adelante utilizarla eso en teoría la puedo llamar herencia. (Si me equivoco corrijan).

Lo intentare porque no me he puesto a observar y practicar mucho la herencia y el poliformismo en la programación.

Saludos;

Novato_erick

novato_erick 04-08-2011 05:55:30

:confused:

Una pregunta Chris:

Código Delphi [-]
type
    TClientesViewer =  class(TBaseViewer);// de donde proviene TBaseViewer es de un componente?

Porque me manda error. identificador no declarado...


Saludos;

roman 04-08-2011 16:20:59

Cita:

Empezado por Chris (Mensaje 408395)
Terminarías repitiendo código Román.

No veo por qué. Se requiere una sóla línea de código para poner en modo de edición o de inserción al dataset. Normalmente, dicho código lo colocaría yo en el evento Execute de una TAction, lo cual te permite llamar al mismo código desde varios lugares. En dicho evento puedes colocar cualquier otra cosa que haga falta para personalizar el visor.

Como dije antes, en mi opinión, es un error de diseño dejar a la clase visora la tarea de escoger ese menester pues, para empezar, dicha clase debería ser agnóstica del dataset.

// Saludos

roman 04-08-2011 16:22:09

Cita:

Empezado por novato_erick (Mensaje 408464)
Es lo que trato de decirles: En teoría son argumentos que exponen los autores en sus reconocidos libros que "Funcionan" sin embargo en la programación me he percatado que es a base de experiencia lo que realmente manda.

Desde luego. Pero muchos de esos autores escriben sus libros basados en su experiencia ;).

// Saludos

novato_erick 04-08-2011 17:09:22

Bueno logre poner en modo de Edición mi procedimiento al llamar el formulario en modo de edición he aquí la forma en que lo deje.
Código Delphi [-]

procedure MuestraVentana(ventana: string; modo: string);

if ventana = 'Clientes' then // lo considero como mi modo 'Normal' al ejecutarlo que significa agregar registros
  begin
    if FrmCliente = nil then
    begin
      FrmCliente := TfrmCliente.create(FrmCliente);
      Try
      FrmCliente.Parent := FrmPrincipal.Panel4;
      FrmCliente.Caption := 'Clientes';
      dmacceso.cdsClientes.Active := True;
      dmacceso.cdsClientes.Insert;
      FrmCliente.Show;
      finally
      If FrmCliente.Caption <> ' ' then
      Begin
      FrmPrincipal.TabSet1.Tabs.add(FrmCliente.Caption);
      FrmPrincipal.TabSet1.TabIndex := FrmPrincipal.TabSet1.Tabs.Count - 1;
      end;
      end;
    end;

    if modo = 'Editar' then
    begin
        frmCliente.btnModificar.Visible := True;// hago que se ponga visible el botón de modificar
        frmCliente.btnGuardar.Visible := False;// aquí oculto el botón de guardar pues esta en modo de edición
        frmCliente := TfrmCliente.Create(nil);
        dmacceso.cdsClientes.Edit;
    end;

  end;

Sin embargo aun pasa algo extraño, no me muestra los datos en seguida en los dbedit al abrir el formulario para ser modificados si no le doy click en algun dbEdit. ¿Alguien tiene alguna idea de como sulucionar este problema en si?

aquí esta como llamo a mi formulario en modo de Edición en el DBGrid con el evento DblClick

Código Delphi [-]
procedure TfrmFicheroCliente.dbgClientesDblClick(Sender: TObject);
begin
 pais := dmAcceso.cdsClientesPais.text;
 MostrarRegistroYGuardar;
end;

aquí llamo el modo de edición con un TButton:

Código Delphi [-]
procedure TfrmFicheroCliente.aModificarExecute(Sender: TObject);
begin
  pais := dmAcceso.cdsClientesPais.text;
  MostrarRegistroYGuardar;
end;

procedure TfrmFicheroCliente.MostrarRegistroYGuardar;
begin
  utiles.MuestraVentana('Clientes', 'Editar');// aquí llama al modo Editar del FrmCliente
end;

oscarac 04-08-2011 17:20:04

mmm me atrevo a preguntar si los dbedits estan asociados a un datasource?

roman 04-08-2011 17:26:02

Yo veo raro tu código. Aparentemente, el código de inserción de un nuevo registro se ejecuta aún cuando lo que quieras sea editar un registro existente. Es más, en tal caso creas la ventana frmCliente ¡dos veces!

Por otra parte, creo que sería mejor mostrar la ventana con ShowModal, a menos que haya alguna razón especial que no conozco.

// Saludos

novato_erick 04-08-2011 17:28:47

Así es oscarac

Este esta en

dmAcceso.cdsClientes

DataSource properties

AutoEdit = True
DataSet = dmAcceso.cdsClientes
Enable = True
Name = dsClientes
Tag = 0

Saludos;

oscarac 04-08-2011 17:33:09

Yo tambien pienso que te estas complicando la existencia.... anteriormente te deje unos ejemplos de como lo hago yo.. quiza no seas una de las mejores formas pero me parece mucho mas sencillo que el tuyo

segun entiendo.. en un formulario de modificacion de datos puedes ejecutar 3 acciones

1.- agregar registros (append)
2.- modificar registros (edit)
3.- eliminar registros (delete)

la informacion en el form en cualquiera de los 3 casosa (con reservas en el modo insertar) es la misma y el procedimiento de aperturar el form deberia ser el mismo

segun YO cuando preguntas si el formulario = nil, es porque no tienes un control del flujo del programa....

creas un formulario y cuando lo cierras lo destruyes.. eso te evita preguntar si esta nil
del mismo modo con los botones... tienes 2 botonos 1 para crear y el otro para modificar... cuando deberias tener solo 1 que puedes cambiar el caption segun la accion y dentro del onclick manejar las "acciones" a seguir... segun hice referencia en los ejemplos que te deje anteriormente

sin animos de desanimarte ni mucho menos de hacerte cambiar de opinion (porque puede existir una buena razon del porque lo haces de esa forma), esos son mis comentarios

salvo error u omision o alguien con mas conocimientos me pueda corregir y de paso aprender mas..

saludos

Chris 04-08-2011 17:40:05

Cita:

Empezado por novato_erick (Mensaje 408467)
:confused:

Una pregunta Chris:

Código Delphi [-]
type
    TClientesViewer =  class(TBaseViewer);// de donde proviene TBaseViewer es de un componente?

Porque me manda error. identificador no declarado...


Saludos;

Lo que sucede es que TBaseViewer no es una clase que venga con Delphi. Es una clase que yo he escrito. Es el ancestro de todos los visores de registro que utilizo en mi aplicación. TBaseViewer implementa la funcionalidad básica y general de un visor de registro.

Saludos,
Chris

roman 04-08-2011 17:42:31

De hecho, ¿por qué frmCliente tiene un botón Modificar y uno Guardar? Yo pondría sólo el botón Guardar que sirve para lo mismo en ambos casos (inserción y edición): guardar los datos. Es más, si el formulario se muestra modalmente, dicho botón Guardar únicamente pondría la propiedad ModalResult del formulario en mrOk para que la ventana que lo abrió sepa que se aceptan los cambios y se haga el Post a la base de datos.

Por otra parte, oscarac, no me queda claro que el mismo formulario se use para borrar registros. Creo que esa operación quedaría mejor en un explorador de registros, siguiendo la terminología de Chris.

// Saludos


La franja horaria es GMT +2. Ahora son las 16:29:40.

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