Projeto de Programação 1 - Programa Controle de Acesso
Proposta de trabalho:
- Sistema de controle de acesso;
- Interface GUI para o administrador;
- Interface usuário GUI simulando o display LCD;
- Arquivo de log de entrada/saída de usuários com horário e data;
- Armazenamento da base de dados em arquivo;
- Prever possibilidade de vários administradores;
Implementações básicas para a Interface GUI
Criando e configurando a janela do Administrador
Aqui pode-se ver as funções básicas da interface gráfica do administrador: criação de uma janela centralizada de tamanho 800x600, própria para a interface, entre outros.
/** Funções para se iniciar e configurar a janela administrador do Controle de Acesso de usuário **/
window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_title(GTK_WINDOW(window), "Controle de Acesso de Usuário - Administrador");
gtk_window_set_default_size(GTK_WINDOW(window), 800, 600);
gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
gtk_container_set_border_width(GTK_CONTAINER(window), 5);
Criando frames para divisão de espaços para a janela
Através do GTK é preciso criar frames para utilizar da forma mais organizada a janela do programa. Assim pelo exemplo abaixo foi criado três frames: (1) Bem Vindo, (2) Opções de Gerenciamento ao administrador e (3) Log de Acesso.
/** Funções para dividir a janela em dois frames: o primeiro (da esquerda) será um campo fixo e o segundo(da direita) conterá as atividades do administrador **/
table1 = gtk_table_new(2, 2, TRUE);
gtk_table_set_row_spacings(GTK_TABLE(table1), 10);
gtk_table_set_col_spacings(GTK_TABLE(table1), 10);
gtk_container_add(GTK_CONTAINER(window), table1);
frame1 = gtk_frame_new("Bem Vindo");
gtk_frame_set_shadow_type(GTK_FRAME(frame1), GTK_SHADOW_IN);
frame2 = gtk_frame_new("Opções de Gerenciamento");
gtk_frame_set_shadow_type(GTK_FRAME(frame2), GTK_SHADOW_IN);
frame3 = gtk_frame_new("Log de acesso");
gtk_frame_set_shadow_type(GTK_FRAME(frame3), GTK_SHADOW_IN);
gtk_table_attach_defaults(GTK_TABLE(table1), frame1, 0, 1, 0, 1);
gtk_table_attach_defaults(GTK_TABLE(table1), frame2, 1, 2, 0, 2);
gtk_table_attach_defaults(GTK_TABLE(table1), frame3, 0, 1, 1, 2);
Botão HOME e mensagen de Boas Vindas
No exemplo abaixo observa-se a implementação de um botão HOME com a função de voltar sempre à página principal quando clicado. Além disso tem-se as funções relacionadas à criação de uma caixa (box) para inserir os botões ordenados das atividades do Admin. Também há uma aplicação básica como mostrar uma mensagem de Boas Vindas.
/** Caixa que conterá itens do menu esquerdo - campo fixo **/
/** Funções para criar o botão HOME **/
/** Mensagem de boas vindas ao usuário **/
vbox2 = gtk_vbox_new(FALSE, 10);
hbox2 = gtk_hbox_new(FALSE, 10);
halign2 = gtk_alignment_new (0,1,0,0);
botaoHome = gtk_button_new_with_label ("Home");
gtk_widget_set_size_request(botaoHome, 70, 35);
boasvindas = gtk_label_new ("Bom Dia Administrador!");
gtk_box_pack_start(GTK_BOX (hbox2), botaoHome, FALSE, TRUE, 10);
gtk_box_pack_start(GTK_BOX (hbox2), boasvindas, FALSE, TRUE, 10);
gtk_container_add (GTK_CONTAINER (halign2), hbox2);
gtk_box_pack_start(GTK_BOX (vbox2), halign2, FALSE, FALSE, 0);
gtk_container_add(GTK_CONTAINER(frame1), vbox2);
/** Função do botão HOME **/
g_signal_connect(G_OBJECT(botaoHome), "clicked", G_CALLBACK(gtk_widget_show_all), NULL);
Funções para fechar a janela e mostrar todos os seus itens
g_signal_connect_swapped(G_OBJECT(window), "destroy", G_CALLBACK(gtk_main_quit), G_OBJECT(window));
/** Função para mostrar todos os itens da janela **/
gtk_widget_show_all(window);
Sistema de controle de acesso
Configuração da janela de Controle de Acesso
No campo abaixo apresentamos a criação de uma tabela que contém 2 linhas e 2 colunas. No lado esquerdo temos a indicação do Usuário e da Senha e do lado direito temos os campos de preenchimento para os dados de entrada.
/** Configurações da estrutura vertical principal da janela **/
vbox = gtk_vbox_new(FALSE, 10);
gtk_container_add(GTK_CONTAINER(janela), vbox);
labelInicio = gtk_label_new ("Entre com seu nome de usuário/administrador e senha para fazer login:");
halignInicio = gtk_alignment_new(0,1,0,0);
gtk_container_add(GTK_CONTAINER(halignInicio), labelInicio);
gtk_box_pack_start(GTK_BOX(vbox), halignInicio, FALSE, FALSE, 10);
/** Configurações da tabela de Login **/
tabLogin = gtk_table_new (3, 3, FALSE);
gtk_box_pack_start (GTK_BOX(vbox), tabLogin, FALSE, FALSE, 10);
labelUsuario = gtk_label_new ("Usuário:");
gtk_table_attach(GTK_TABLE(tabLogin), labelUsuario, 0, 1, 0, 1, GTK_FILL | GTK_SHRINK, GTK_FILL | GTK_SHRINK, 5, 5);
labelSenha = gtk_label_new ("Senha:");
gtk_table_attach(GTK_TABLE(tabLogin), labelSenha, 0, 1, 1, 2, GTK_FILL | GTK_SHRINK, GTK_FILL | GTK_SHRINK, 5, 5);
/**Campo para entrada de nome do usuário/administrador**/
entryUser = gtk_entry_new ();
gtk_table_attach(GTK_TABLE(tabLogin), entryUser, 1, 2, 0, 1, GTK_FILL | GTK_SHRINK, GTK_FILL | GTK_SHRINK, 5, 5);
/**Campo para entrada de senha do usuário/administrador**/
entrySenha = gtk_entry_new ();
gtk_entry_set_visibility(GTK_ENTRY (entrySenha), FALSE);
gtk_table_attach(GTK_TABLE(tabLogin), entrySenha, 1, 2, 1, 2, GTK_FILL | GTK_SHRINK, GTK_FILL | GTK_SHRINK, 5, 5);
/**Botão "OK" para confirmar login**/
buttonOK = gtk_button_new_with_label ("OK");
gtk_widget_set_size_request(buttonOK, 35, 25);
gtk_table_attach(GTK_TABLE(tabLogin), buttonOK, 2, 3, 1, 2, GTK_FILL | GTK_SHRINK, GTK_FILL | GTK_SHRINK, 5, 5);
gtk_widget_show_all(janela);
g_signal_connect(G_OBJECT(buttonOK), "clicked", G_CALLBACK(buttonOK_clicked), NULL);
Função de validação de entrada
Funções básicas para o Controle de Acesso do usuário como validação dos dados de entrada.
/** Função que valida um usuário e abre a porta **/
void tratar_usuario(char *userID, char *senha)
{
int userEncontrado=1;
int i=0;
for (i=0;i<TAB_USER && userEncontrado; i++) {
if( strcmp(userID, TabelaUsuarios[i].UserId)==0)
userEncontrado=0;
}
/* Se usuário encontrado abre a porta */
if (userEncontrado==0) {
i--; /* o indice do sujeito é i-1 */
if (TabelaUsuarios[i].ContUser>=3) {
printf("\nUsuário Bloqueado! Contate o administrador para desbloquear o usuário.\n");
}else {
if(TabelaUsuarios[i].ContUser<3){
if(strcmp(senha,TabelaUsuarios[i].Senha)==0) {
TabelaUsuarios[i].ContUser=0;
printf("\nAbrir porta!!!");
printf("\n%s\n%s %s!\n", MsgPadrao, TabelaUsuarios[i].MsgUser, userID);
registrar_login(userID);
}else {/*Bloquear usuário se a senha dada é errada mais de três vezes*/
zerar();
TabelaUsuarios[i].ContUser++;
if(TabelaUsuarios[i].ContUser<3)
printf("Senha Inválida. Tente novamente:\n");
else
printf("Senha Inválida!\n");
}
}
if(TabelaUsuarios[i].ContUser>=3)
printf("\nUsuário Bloqueado!!!\n");
}
}else
printf("\nUsuário requerido não cadastrado.\n");
}
Funções de Login de Usuário
Abaixo temos as funções relacionadas ao tratamento de dados de entrada do usuário. Ao extrair os dados dos campos de preenchimento da interface, realiza-se uma conferência com as dados já cadastrados nas listas de usuários. Então só depois é possível o acesso do usuário já cadastrado.
/*Funções de Login de usuário e/ou admin*/
stat ic void buttonOK_clicked(GtkWidget *widget, gpointer data)
{
int i=0, aux=0, adminEncontrado=1, userEncontrado=1;
CCusuario = gtk_entry_get_text(GTK_ENTRY (entryUser));
CCsenha = gtk_entry_get_text(GTK_ENTRY (entrySenha));
userID = g_strdup (CCusuario);
senha = g_strdup (CCsenha);
g_printf("\nUsuário=%s\n", userID);
g_printf("\nSenha=%s\n", senha);
/*Analisar no campo de login se é admin, e chamar função de tratamento para admin*/
for (aux=0;aux<TAB_ADMIN && adminEncontrado; aux++) {
if( strcmp(userID, TabelaAdmin[aux].UserAdmin)==0)
adminEncontrado=0;
}
if (adminEncontrado==0) {
gtk_main_quit();
tratar_admin(userID, senha);
}
/*Analisar no campo de login se é usuário comum, e chamar função de tratamento para usuário*/
for (i=0;i<TAB_USER && userEncontrado; i++) {
if( strcmp(userID, TabelaUsuarios[i].UserId)==0)
userEncontrado=0;
}
if (userEncontrado==0) {
gtk_main_quit();
tratar_usuario(userID, senha);
}
gtk_main();
Função zerar campo de preenchimento
Com essa função, apaga-se a senha errada que o próprio usuário fornece. Logo é dada mais uma alternativa para o usuário digitar e fornecer a senha novamente.
/*Função para apagar a senha erroneamente digitada no campo de senha*/
void zerar()
{
gtk_editable_delete_text( GTK_EDITABLE(entrySenha), 0, -1 );
}
Ações do Administrador
Bloquear Usuário
case 1: /*** PARA BLOQUEAR USUÁRIO ***/
printf("\nPara bloquear usuário:\n");
printf("\nEntre com o nome do usuário a ser bloqueado: ");
scanf(" %s", userID);
/* Loop para encontrar o usuário na tabela */
for (aux=0, userEncontrado=1;aux<TAB_USER && userEncontrado;aux++) {
if( strcmp(userID, TabelaUsuarios[aux].UserId)==0)
userEncontrado=0;
}
/* Se o usuário for encontrado será bloqueado */
if (userEncontrado==0) {
aux--;
if (TabelaUsuarios[aux].ContUser>=3) {
printf("\nUsuário encontrado já bloqueado!\n");
} else{
printf("\nBloqueando usuário...\n");
TabelaUsuarios[aux].ContUser = 3;
/* Condição para verificar o sucesso da operação.*/
if(TabelaUsuarios[aux].ContUser>=3)
printf("\nUsuário bloqueado com sucesso!\n");
}
} else
printf("\nUsuário não cadastrado!\n");
break;
Cadastrar novo Usuário
case 2: /*** PARA CADASTRAR NOVO USUÁRIO ***/
printf("\nPara cadastrar novo usuário: ");
/* Loop para encontrar o final da tabela */
for(i=0;TabelaUsuarios[i].UserId[0] && i<TAB_USER;i++)
aux++;
/* Condição para novo cadastro.*/
if (i==TAB_USER)
printf("\nLimite de cadastro de usuários atingido.\n");
else {
printf("\nEntre com um novo nome de Usuário:\n");
scanf(" %s", userID);
/* Loop para encontrar o usuário na tabela */
for (i=0; i<aux && userEncontrado; i++) {
if( strcmp(userID, TabelaUsuarios[i].UserId)==0)
userEncontrado=0;
}
/* Se o usuário for encontrado já está cadastrado. */
if (userEncontrado==0)
printf("\nUsuário já cadastrado.Tente novamente com outro usuário.\n");
else {
strcat(TabelaUsuarios[aux].UserId,userID);
printf("\nEntre com uma nova senha para o novo usuário %s:\n", userID);
scanf(" %s", senha);
strcat(TabelaUsuarios[aux].Senha,senha);
strcat(TabelaUsuarios[aux].MsgUser,"Bom Dia");
/* Condição para verificar o sucesso da operação.*/
if(strcmp(userID, TabelaUsuarios[aux].UserId)==0 && strcmp(senha, TabelaUsuarios[aux].Senha)==0)
printf("\nNovo usuário '%s' cadastrado com sucesso!\n", TabelaUsuarios[aux].UserId);
}
}
break;
Desbloquear um Usuário
case 3: /*** PARA DESBLOQUEAR USUÁRIO ***/
printf("\nPara desbloquear usuário:\n");
printf("\nEntre com o nome do usuário a ser desbloqueado: ");
scanf(" %s", userID);
/* Loop para encontrar o usuário na tabela */
for (aux=0;aux<TAB_USER && userEncontrado;aux++) {
if( strcmp(userID, TabelaUsuarios[aux].UserId)==0)
userEncontrado=0;
}
/* Se o usuário for encontrado será desbloqueado */
if (userEncontrado==0) {
aux--;
if (TabelaUsuarios[aux].ContUser>=3) {
printf("\nUsuário encontrado bloqueado! Desbloqueando usuário...");
TabelaUsuarios[aux].ContUser=0;
/* Condição para verificar o sucesso da operação.*/
if(TabelaUsuarios[aux].ContUser<=3)
printf("\nUsuário desbloqueado com sucesso!\n");
} else
printf("\nUsuário encontrado desbloqueado! Tente novamente com outro usuário.\n");
} else
printf("\nUsuário não cadastrado!\n");
break;
Excluir um Usuário
case 4: /*** PARA EXCLUIR UM USUÁRIO ***/
printf("\nDigite o nome de usuário que deseja excluir: ");
scanf(" %s", userID);
/* Loop para encontrar o usuário na tabela.*/
for (i=0;i<TAB_USER && userEncontrado; i++) {
if( strcmp(userID, TabelaUsuarios[i].UserId)==0)
userEncontrado=0;
}
/* Se usuário for encontrado será deletado */
if (userEncontrado==0) {
printf("\nExcluindo usuário...\n");
i--;
TabelaUsuarios[i].UserId[0]='\0';
TabelaUsuarios[i].Senha[0]='\0';
TabelaUsuarios[i].ContUser=0;
TabelaUsuarios[i].MsgUser[0]='\0';
/* Condição para verificar o sucesso da operação.*/
if(TabelaUsuarios[i].UserId[0]==0)
printf("\nUsuário excluído com sucesso!\n");
} else
printf("\nUsuário não cadastrado!\n");
break;
Imprimir Tabela de Usuários
case 5: /*** PARA IMPRIMIR TABELA DE USUÁRIOS. ***/
printf("\nListando tabela de usuários cadastrados...\n");
for(aux=0;TabelaUsuarios[aux].UserId[0] && aux<TAB_USER;aux++) {
printf("\n%s",TabelaUsuarios[aux].UserId);
cont++;
}
printf("\n\nNúmero de usuários cadastrados: %d\n", cont);
break;
Limpar Lista de Usuários
case 6: /*** PARA LIMPAR LISTA DE USUÁRIOS ***/
printf("\nLimpando tabela de usuários...\n");
for(i=0;TabelaUsuarios[i].UserId[0];i++) {
TabelaUsuarios[i].UserId[0]='\0';
TabelaUsuarios[i].Senha[0]='\0';
TabelaUsuarios[i].ContUser=0;
TabelaUsuarios[i].MsgUser[0]='\0';
}
break;
Modificar mensagem particular do Usuário
case 7: /*** PARA MODIFICAR MENSAGEM PARTICULAR DO USUÁRIO. ***/
printf("\nPara modificar mensagem particular do usuário:\n");
printf("\nEntre com o nome de usuário que deseja modificar a mensagem: ");
scanf(" %s", userID);
/* Loop para encontrar o usuário na tabela.*/
for (i=0;i<TAB_USER && userEncontrado; i++) {
if( strcmp(userID, TabelaUsuarios[i].UserId)==0)
userEncontrado=0;
}
/* Se usuário for encontrado será modificado a mensagem particular. */
if (userEncontrado==0) {
printf("\nEntre com a nova mensagem: ");
i--;
scanf(" %s", TabelaUsuarios[i].MsgUser);
/* Verificação do sucesso da operação.*/
printf("\nA nova mensagem é '%s'.\n", TabelaUsuarios[i].MsgUser);
printf("\nMensagem particular modificada com sucesso!\n");
}else
printf("\nUsuário não cadastrado!\n");
break;
Modificar mensagem padrão
case 13: /*** PARA MODIFICAR MENSAGEM PADRAO ***/
printf("\nPara modificar mensagem padrão:\n");
printf("\nEntre com a nova mensagem: ");
scanf(" %s", MsgPadrao);
/* Verificação do sucesso da operação.*/
printf("\nA nova mensagem é '%s'.\n", MsgPadrao);
printf("\nMensagem padrão modificada com sucesso!\n");
break;
Modificar senha do Administrador
case 15: /*** PARA MODIFICAR SENHA DO ADMINISTRADOR ***/
printf("\nPara modificar senha do administrador:\n");
printf("\nEntre com a senha atual do admin: ");
scanf(" %s", senha);
for (i=0;i<TAB_ADMIN && userEncontrado; i++) {
if( strcmp(userID, TabelaAdmin[i].UserAdmin)==0)
userEncontrado=0;
}
/* Condição para modificar a atual senha do administrador.*/
if (userEncontrado==0) {
i--;
if (strcmp(senha, TabelaAdmin[i].SenhaAdmin)==0) {
printf("\nEntre com a nova senha: ");
scanf(" %s", senha);
printf("\nConfirme a nova senha: ");
scanf(" %s", TabelaAdmin[i].SenhaAdmin);
/* Condição para verificar o sucesso da operação.*/
if (strcmp(senha, TabelaAdmin[i].SenhaAdmin)==0)
printf("\nSenha do administrador modificada com sucesso!\n");
else
printf("\nFalha ao modificar a senha. Dados incorretos. Tente novamente e verifique a nova senha.\n");
} else
printf("\nDados incorretos. Tente novamente e verifique a senha do administrador fornecida.\n");
}
break;
Interface do Administrador (GUI)
Chamada de funções para Administrar
Abaixo encontra-se um exemplo de chamada de funções do GTK para acessar opções de atividades do Administrador:
/** Chamadas funções para usuário **/
void BloquearUser_clicked(GtkWidget *widget, gpointer data)
{
administrar(1, userID, senha);
}
/** Chamadas funções para administrador **/
void BloquearAdmin_clicked(GtkWidget *widget, gpointer data)
{
administrar(8, userID, senha);
}
Tabela contendo os botões de atividades do Administrador
Tais botões estarão chamando os tipos de funções mostrados acima e são criados seguindo o exemplo abaixo:
/** Caixa que conterá as atividades do administrador **/
vbox1 = gtk_vbox_new(FALSE, 1);
labelopcoes = gtk_label_new("Entre com uma das opções:");
gtk_label_set_justify(GTK_LABEL(labelopcoes), GTK_JUSTIFY_CENTER);
labeluser = gtk_label_new("Para usuário:");
labeladmin = gtk_label_new("Para administrador:");
/** Definição dos textos que representam cada botão de ação **/
BloquearUser = gtk_button_new_with_label("Bloquear usuário");
BloquearAdmin = gtk_button_new_with_label("Bloquear administrador");
/** Criação dos botões para uso nas atividades do admin **/
gtk_box_pack_start(GTK_BOX(vbox1), labelopcoes, 0, TRUE, 0);
gtk_box_pack_start(GTK_BOX(vbox1), labeluser, 0, TRUE, 0);
gtk_box_pack_start(GTK_BOX(vbox1), BloquearUser, 0, TRUE, 0);
gtk_box_pack_start(GTK_BOX(vbox1), CadastrarUser, 0, TRUE, 0);
/* Função do botão "Bloquear usuário"*/
g_signal_connect(G_OBJECT(BloquearUser), "clicked", G_CALLBACK(BloquearUser_clicked), NULL);
/* Função do botão "Bloquear administrador"*/
g_signal_connect(G_OBJECT(BloquearAdmin), "clicked", G_CALLBACK(BloquearAdmin_clicked), NULL);
Log de Acesso
Salvar Log em arquivo de texto no computador
struct TRegistroLog TabelaLog[TAB_LOG];
/*Função para salvar o log de entrada e/ou saída em um arquivo de texto no computador*/
void save()
{
FILE *log;
int i;
register int aux;
log=fopen("LogAcesso.txt", "w");
if (log==NULL) {
printf("\nO arquivo não pode ser aberto.");
return;
}
for(i=0;TabelaLog[i].Usuario[0] && i<TAB_LOG;i++)
fprintf(log, "%s %s %s", TabelaLog[i].Usuario, TabelaLog[i].Entrada, TabelaLog[i].Saida);
fclose(log);
}
Abrir o arquivo texto de Log
A função abaixo indica como salvar as informações de uma estrutura em arquivo de texto.
/*****FUNÇÃO NÃO IMPLEMENTADA, PORÉM PARCIALMENTE PRONTA, PARA ABRIR O ARQUIVO DE LOG PELO ADMINISTRADOR*****/
void load ()
{
FILE *log;
register int i;
log=fopen("LogAcesso.txt", "r");
if (log==NULL)
prinf("\nO arquivo não pode ser aberto.");
return;
}
init_list();
for (i=0; i<TAB_LOG; i++)
if (fread(&TabelaLog[i], sizeof (struct TRegistroLog), 1, log)!=1) {
if (feof (log)) break;
printf("\nErro de escrita no arquivo de log.\n");
}
fclose(log);
}
Função para registrar o login
No exemplo abaixo é apresentado os passos para registrar os dados de entrada do usuário no exato momento de entrada do mesmo.
/*Função para guardar a informação de entrada de um usuário numa lista de log que será mostrada ao administrador quando ele estiver logado*/
void registrar_login(char *userID)
{
int i=0, aux=0;
int algoEncontrado=1;
char *chamadaLog;
time_t lt;
lt = time(NULL);
chamadaLog = ctime(<);
/* Loop para encontrar se há algo na tabela de log */
for(i=0;TabelaLog[i].Usuario[0] && i<TAB_LOG;i++)
aux++;
/* Adicionando registro de log */
strcat(TabelaLog[aux].Usuario, userID);
strcat(TabelaLog[aux].Entrada, chamadaLog);
/* Condição para verificar o sucesso da operação.*/
if(strcmp(userID, TabelaLog[aux].Usuario)==0 && strcmp(chamadaLog, TabelaLog[aux].Entrada)==0)
printf("\nEntrada do usuário %s em %s\n\n", TabelaLog[aux].Usuario, TabelaLog[aux].Entrada);
save();
}
Função para registrar o logout
Aqui temos o modelo básico de implementação do registro de dados de saída do usuário no exato momento de saída do mesmo.
/*****FUNÇÃO NÃO IMPLEMENTADA, PORÉM PARCIALMENTE PRONTA, PARA GUARDAR A INFORMAÇÃO DE SÁIDA DE UM USUÁRIO NA LISTA DE LOG TAMBÉM MOSTRADA AO ADMINISTRADOR*****/
void registrar_logout(char *userID)
{
time_t ltout;
TabelaLog.Usuario = userID;
ltout = time(NULL);
TabelaLog.Saida = ctime(<out);
}
Definição dos itens da Lista de Log
/***Definição dos itens na lista de log***/
enum
{
NAME_COLUMN,
ENTRY_COLUMN,
OUT_COLUMN,
N_COLUMNS
};
const gchar *nome, *in, *out;
Criação da Lista de Log
/***Função de criação da lista de log***/
static void
init_list(GtkWidget *list)
{
GtkListStore *store;/*Tipo de widget para criar uma lista(modelo) inicialmente não visualizada*/
GtkTreeViewColumn *column;/*Tipo de widget para visualizar a lista já criada */
GtkCellRenderer *renderer;
/* Criar um novo título na tabela de log "cell_renderer" */
renderer = gtk_cell_renderer_text_new ();
/* Criar uma coluna, associando o próximo atributo do "cell_renderer" com a primeira coluna do modelode lista */
column = gtk_tree_view_column_new_with_attributes ("Nome", renderer, "text", NAME_COLUMN, NULL);
/* Adicionar a coluna criada ao campo de visualização */
gtk_tree_view_append_column (GTK_TREE_VIEW (list), column);
/* Segunda coluna ...*/
renderer = gtk_cell_renderer_text_new ();
column = gtk_tree_view_column_new_with_attributes ("Entrada", renderer, "text", ENTRY_COLUMN, NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW (list), column);
/* Última Coluna ...*/
renderer = gtk_cell_renderer_text_new ();
column = gtk_tree_view_column_new_with_attributes ("Saída", renderer, "text", OUT_COLUMN, NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW (list), column);
/* Criação do modelo de lista. Uso do "store model"*/
store = gtk_list_store_new (3, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
gtk_tree_view_set_model(GTK_TREE_VIEW(list),
GTK_TREE_MODEL(store));
g_object_unref(G_OBJECT (store));
}
Composição da Lista de Log
/***Função de composição da lista de log***/
static void
add_to_list(GtkWidget *list)
{
GtkListStore *store;
GtkTreeIter iter;
nome = TabelaLog[i].Usuario;
in = TabelaLog[i].Entrada;
out = TabelaLog[i].Saida;
store = GTK_LIST_STORE(gtk_tree_view_get_model
(GTK_TREE_VIEW(list)));
gtk_list_store_insert_after(store, &iter, NULL);
gtk_list_store_set(store, &iter, NAME_COLUMN, nome, -1);
gtk_list_store_set(store, &iter, ENTRY_COLUMN, in, -1);
gtk_list_store_set(store, &iter, OUT_COLUMN, out, -1);
/** Frame 3 com a lista de log - campo fixo **/
list = gtk_tree_view_new();
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(list), TRUE);
gtk_container_add(GTK_CONTAINER(frame3), list);
init_list(list);
for(i=0;TabelaLog[i].Usuario[0] && i<TAB_LOG;i++)
add_to_list(list);