public static Factura InsertFactura(Factura factura, SqlTransaction transaction)
        {
            var param = new List<SPParameter>
                {
                    new SPParameter("Numero", factura.Numero),
                    new SPParameter("Fecha", factura.Fecha),
                    new SPParameter("Total", factura.Total),
                    new SPParameter("ID_Forma_Pago", factura.FormaPago.ID),
                    new SPParameter("ID_Usuario" , factura.Usuario.ID)
                };

            var sp = (transaction != null)
                        ? new StoreProcedure(DataBaseConst.Factura.SPInsertFactura, param, transaction)
                        : new StoreProcedure(DataBaseConst.Factura.SPInsertFactura, param);

            factura.ID = (int)sp.ExecuteScalar(transaction);

            return factura;
        }
        private void LblFacturar_Click(object sender, EventArgs e)
        {
            #region Validaciones

            var exceptionMessage = string.Empty;

            if (TypesHelper.IsEmpty(TxtCantidad.Text) || !TypesHelper.IsNumeric(TxtCantidad.Text))
                exceptionMessage = "Ingrese una cantidad correcta de publicaciones a facturar";

            if (PagaConTarjeta)
            {
                #region Validaciones datos tarjeta

                if (TypesHelper.IsEmpty(TxtCodSeguridad.Text))
                    exceptionMessage += Environment.NewLine + "El codigo de seguridad de la tarjeta es obligatorio";

                if (!TypesHelper.IsNumeric(TxtCodSeguridad.Text))
                    exceptionMessage += Environment.NewLine + "El codigo de seguridad de la tarjeta debe ser numerico";

                if (TypesHelper.IsEmpty(TxtDni.Text))
                    exceptionMessage += Environment.NewLine + "El numero de documento del titular de la tarjeta es obligatorio";

                if (!TypesHelper.IsNumeric(TxtDni.Text))
                    exceptionMessage += Environment.NewLine + "El numero de documento del titular de la tarjeta debe ser numerico";

                if (TypesHelper.IsEmpty(TxtNroTarjeta.Text))
                    exceptionMessage += Environment.NewLine + "El numero de la tarjeta es obligatorio";

                if (!TypesHelper.IsNumeric(TxtNroTarjeta.Text))
                    exceptionMessage += Environment.NewLine + "El numero de la tarjeta debe ser numerico";

                if (TypesHelper.IsEmpty(TxtTarjeta.Text))
                    exceptionMessage += Environment.NewLine + "La descripcion de la tarjeta es obligatorio";

                if (TypesHelper.IsEmpty(TxtTitular.Text))
                    exceptionMessage += Environment.NewLine + "El nombre del titular de la tarjeta es obligatorio";

                if (TypesHelper.IsEmpty(TxtVencimiento.Text))
                    exceptionMessage += Environment.NewLine + "La fecha de vencimiento de la tarjeta es obligatorio";

                if (!TypesHelper.IsNumeric(TxtVencimiento.Text))
                    exceptionMessage += Environment.NewLine + "La fecha de vencimiento de la tarjeta debe ser numerico";

                #endregion
            }

            if (!TypesHelper.IsEmpty(exceptionMessage))
            {
                MessageBox.Show(exceptionMessage, "Atencion");
                return;
            }

            #endregion

            #region Obtengo las publicaciones que voy a rendir

            var cant = Convert.ToInt32(TxtCantidad.Text.Trim());
            if (cant > LstPublicaciones.Items.Count)
                cant = LstPublicaciones.Items.Count;

            var listaPublicacionesAFacturar = PublicationsList.GetRange(0, cant);
            var itemsFactura = new List<ItemFactura>();

            #endregion

            #region Obtengo todos los items de la factura

            foreach (var publicacionFacturar in listaPublicacionesAFacturar)
            {
                var comprasPorPublicacion = CompraPersistance.GetByPublicationId(publicacionFacturar.ID);
                if (comprasPorPublicacion != null)
                {
                    //Pueden ser tanto compras inmediatas o subastas, esta hecho de forma polimorfica
                    foreach (var compra in comprasPorPublicacion)
                        itemsFactura.Add(compra.ConvertToItemFactura());
                }

                //Agrego el item factura de la publicacion en si misma (aunque no tenga compras)
                itemsFactura.Add(publicacionFacturar.ConvertToItemFactura());
            }

            #endregion

            using (var transaction = DataBaseManager.Instance().Connection.BeginTransaction(IsolationLevel.Serializable))
            {
                try
                {
                    var dialogAnswer = MessageBox.Show("¿Está seguro que quiere facturar " + cant + " publicacion/es?", "Atención", MessageBoxButtons.YesNo);
                    if (dialogAnswer == DialogResult.Yes)
                    {
                        #region Armo y guardo la factura en si misma

                        var factura = new Factura
                        {
                            Fecha = ConfigurationVariables.FechaSistema,
                            Numero = FacturaPersistance.GetUltimoNumeroFactura(transaction) + 1,
                            FormaPago = FormaPagoPersistance.GetById((int)CboFormaPago.SelectedValue, transaction),
                            Usuario = (Usuario)SessionManager.CurrentUser,
                            Total = itemsFactura.Sum(item => item.Monto)
                        };

                        //Guardo la factura en la base de datos (y me retorna la factura insertada con el ID correspondiente)
                        factura = FacturaPersistance.InsertFactura(factura, transaction);

                        #endregion

                        if (factura.ID != 0)
                        {
                            Dictionary<int, int> visibilidadesPorUsuario = VisibilidadPersistance.GetAllByUser(SessionManager.CurrentUser, transaction);

                            #region Guardo los items en la base

                            //Creación de los Items de la Factura
                            foreach (var item in itemsFactura)
                            {
                                item.Factura = factura;

                                #region Logica de bonificacion de items

                                /* Realizo la modificacion del contador solo si el item generado surgio a partir
                                de una compra y no es el item de la publicacion en si misma. */
                                if (item.ContadorBonificacion)
                                {
                                    var idVisibilidad = item.Publicacion.Visibilidad.ID;

                                    if (visibilidadesPorUsuario.ContainsKey(idVisibilidad))
                                        visibilidadesPorUsuario[idVisibilidad]++;
                                    else
                                        visibilidadesPorUsuario.Add(idVisibilidad, 1);

                                    if (visibilidadesPorUsuario[idVisibilidad] == 10)
                                        item.Monto = visibilidadesPorUsuario[idVisibilidad] = 0;
                                }

                                #endregion
                            }

                            var insertoCorrectamente = ItemFacturaPersistance.InsertItemsFactura(itemsFactura, transaction);

                            if (!insertoCorrectamente)
                                throw new Exception("Se produjo un error durante la insercion de los items de la factura");

                            #endregion

                            #region Guardo la informacion de las visibilidades

                            insertoCorrectamente = VisibilidadPersistance.InsertVisibilidadesRendidasPorUsuario(visibilidadesPorUsuario, SessionManager.CurrentUser.ID, transaction);
                            if (!insertoCorrectamente)
                                throw new Exception("Se produjo un error durante la insercion de la informacion de las visibilidades rendidas por usuario");

                            #endregion

                            #region Guardo la informacion de la tarjeta de credito

                            if (PagaConTarjeta)
                            {
                                var tarjeta = new TarjetaCredito
                                {
                                    Tarjeta = TxtTarjeta.Text,
                                    NumeroTarjeta = Convert.ToInt32(TxtNroTarjeta.Text),
                                    Vencimiento = Convert.ToInt32(TxtVencimiento.Text),
                                    CodigoSeguridad = Convert.ToInt32(TxtCodSeguridad.Text),
                                    Titular = TxtTitular.Text,
                                    DniTitular = Convert.ToInt32(TxtDni.Text),
                                    Factura = factura
                                };

                                tarjeta = TarjetaCreditoPersistance.Insert(tarjeta, transaction);

                                if (tarjeta.ID == 0)
                                    throw new Exception("Se produjo un error durante la insercion de la informacion de la tarjeta");
                            }

                            #endregion

                            transaction.Commit();

                            MessageBox.Show("Se facturaron satisfactoriamente las publicaciones", "Atencion");
                            CompleteAction = true;

                            Close();
                        }
                        else
                            transaction.Rollback();
                    }
                }
                catch (Exception ex)
                {
                    transaction.Rollback();

                    TxtCantidad.Clear();
                    MessageBox.Show(ex.Message, "Atencion");
                }
            }
        }