Quem j?? foi em supermercado e nunca viu aquelas maquininhas que o operador fica digitando? Pois ??… estamos falando dela mesma. Parece ser f??cil seu uso, mas quem come??a a trabalhar com elas ?? de perder os cabelos.
Nesse artigo veremos como comunicar com um microterminal via TCP/IP e trocar informa????es (ida e volta). Para isso vamos utilizar o modelo da Gertec MT 720. Uma breve descri????o do aparelho e seu uso (retirado daqui):
“O MT 720 ?? um excelente microterminal que opera em rede Ethernet com protocolos TCP-IP. Seu teclado com 20 teclas program??veis pode ser utilizado com fun????es espec??ficas de atalhos para agilizar a opera????o.
As teclas re-legend??veis permitem a identifica????o da fun????o de cada tecla. O display com back light (iluminado), com 20 caracteres por 2 linhas, permite facilmente a visualiza????o da informa????o, principalmente em ambientes que possuem pouca ilumina????o, como bares noturnos, por exemplo.
Atrav??s de uma porta de teclado (AT/PS2) e mais 3 portas Seriais (RS-232) ?? poss??vel a conex??o de perif??ricos como: Leitor de C??digo de Barras, Display de Cliente, Balan??a, Impressora, etc.
O MT 720 ?? ideal para aplica????es de Cart??o de Consumo, solu????es para Bares, Lanchonetes, Restaurantes Self-Service, Livrarias, Papelarias, apontamento de produ????o em Ind??strias.”
Na pr??pria p??gina do dispositivo (citada acima) tem alguns downloads de arquivos, dentre eles alguns c??digos-fontes com exemplos de como fazer a comunica????o bem como realizar as opera????es pelas quais foi criado. Um por??m: s?? tem dispon??veis nas linguagens C++, Delphi e VB (n??o ?? VB.NET).
Da?? pensamos: “sem problemas! ?? s?? converter pra VB.NET a partir do VB”… Tamb??m pensei nisso e n??o consegui! Dando uma olhada de perto no c??digo-fonte d?? para saber que as proezas encontradas s??o s?? poss??veis por artif??cios que a linguagem oferece: por exemplo, ponteiros. Quem j?? programou em VB via a grande flexibilidade da linguagem de fazer tudo o que quer e do jeito que era mais conveniente. Maravilha! Pena que tenhamos que seguir certos padr??es. Fica a dica: se n??o quiser perder tempo, pegue logo o c??digo prontinho l?? na p??gina do fabricante (vem tamb??m no CD quando compra o aparelho) e utilize. De prefer??ncia use o Delphi e em seguida em VB. Se quiser aventurar vamos com C#…
Se escolheu C# siga lendo pois, em diante, veremos como fazer… Teremos os seguintes objetivos na implementa????o:
- Criar uma ??nica aplica????o que ir?? receber e enviar os dados;
- Exibir o que foi digitado tanto no microterminal quanto no monitor.
Instala????o
Leia o manual do usu??rio aqui. Em resumo: ligue o microterminal na rede el??trica e conecte na porta traseira um cabo RJ-45 no dispositivo e a rede. Em seguida configure, no microterminal, o IP do Servidor (no meu caso, considerei minha m??quina local com o IP 192.168.1.100, por exemplo). Demais configura????es a seu gosto…
Implementa????o
Criemos um Windows Form. Coloquei alguns controles para embelezar a aplica????o…
O mais importante ?? adicionar um objeto Timer. Os demais coloquei para gerar a aplica????o final (que ficar?? no tray – perto do rel??gio – e outras coisas fru-frus). Configure seu Timer com Interval de 50. Pegue a DLL pmtg.dll e copie para sua aplica????o. Eu disse copie! Voc?? n??o conseguir?? aplicar Reference sobre ela. Clique com o bot??o direito sobre a DLL e escolha Properties. Em Copy to Output Directory coloque Copy Always.
Agora vamos ao c??digo! Clique sobre o WinForm e escolha View Code. Cole os c??digos abaixo para que possamos us??-las no form:
// VARI??VEIS
IntPtr v_Hwnd;
int statusCard;
int statusSerial;
int MsgReceiveData = 1;
string[] dados = new string[255];
// INICIALIZA????O DE FUN????ES EXTERNAS
[DllImport(“pmtg.dll”, CallingConvention = CallingConvention.StdCall, SetLastError = true)]
public static extern int mt_startserver(IntPtr mywhnd, int conecmsg, int commumsg);
[DllImport(“pmtg.dll”, CallingConvention = CallingConvention.StdCall, SetLastError = true)]
public static extern void mt_finishserver();
[DllImport(“pmtg.dll”, CallingConvention = CallingConvention.StdCall, SetLastError = true)]
public static extern int mt_getkey(int id, StringBuilder str);
[DllImport(“pmtg.dll”, CallingConvention = CallingConvention.StdCall, SetLastError = true)]
public static extern int mt_backspace(int id);
[DllImport(“pmtg.dll”, CallingConvention = CallingConvention.StdCall, SetLastError = true)]
public static extern int mt_dispstr(int id, string str);
[DllImport(“pmtg.dll”, CallingConvention = CallingConvention.StdCall, SetLastError = true)]
public static extern int mt_dispch(int id, char ch);
[DllImport(“pmtg.dll”, CallingConvention = CallingConvention.StdCall, SetLastError = true)]
public static extern int mt_dispclrln(int id, int lin);
[DllImport(“pmtg.dll”, CallingConvention = CallingConvention.StdCall, SetLastError = true)]]
public static extern int mt_gotoxy(int id, int lin, int col);
Em seguida adicionamos as duas fun????es primordiais: in??cio (quando carregar o formul??rio) e t??rmino (quando fechar o formul??rio):
private void MicroTerminal_Load(object sender, EventArgs e)
{
v_Hwnd = this.Handle;
statusCard = 0;
statusSerial = 0;
if (mt_startserver(v_Hwnd, 0, MsgReceiveData) == 1)
{
temporizadorLista.Enabled = true;
}
else
MessageBox.Show(“N??o foi poss??vel carregar os m??dulos de conex??o ?? leitora!”);
}
private void MicroTerminal_FormClosed(object sender, FormClosedEventArgs e)
{
temporizadorLista.Enabled = false;
mt_finishserver();
}
Lembrando que esses eventos devem estar configurados (n??o ?? apenas copiar e colar o c??digo, lembra? deve-se indicar l?? em Events do Form). Com isso, ao fazer o primeiro teste, sua aplica????o j?? estar?? recebendo dados do microterminal. Em seguida devemos adicionar um evento Tick ao Timer. Ou seja, a cada 50 milisegundos (lembra que colocamos com esse valor) ir?? ser executado um determinado m??todo.
Antes disso vamos criar um m??todo que ir?? limpar a tela do microterminal e reiniciar os dados de entrada:
private void ReiniciaEntrada(int i)
{
// Limpa as duas linhas e o que foi armazenado
dados[i] = “”;
mt_dispclrln(i, 1);
mt_dispclrln(i, 2);
// Coloca o cursor na primeira linha e prepara o display
mt_gotoxy(i, 1, 0);
mt_dispstr(i, “Numero: “);
mt_gotoxy(i, 2, 0);
}
Enfim adicionamos o c??digo que ir?? ler temporariamente cada leitora conectada ?? rede e coletar/processar os dados de acordo com o que desejar. No meu caso deixei o seguinte:
private void temporizadorLista_Tick(object sender, EventArgs e)
{
StringBuilder rntStr = new StringBuilder();
int i = 0;
// Loop em toda faixa de Ips ??teis: cada leitora conectada na rede
while (i < 255)
{
// Captura algo do teclado
if (mt_getkey(i, rntStr) == 1)
{
string chrAsHex = ((int)rntStr[0]).ToString(“x”);
// Entrada do display caso seja um n??mero
if ((chrAsHex == “3030”) || (chrAsHex == “30”) || (chrAsHex == “31”) || (chrAsHex == “32”) || (chrAsHex == “33”) || (chrAsHex == “34”) || (chrAsHex == “35”) || (chrAsHex == “36”) || (chrAsHex == “37”) || (chrAsHex == “38”) || (chrAsHex == “39”))
{
if (dados[i].Length >= 20)
dados[i] = dados[i].Substring(0, 19) + rntStr[0];
else
dados[i] += rntStr[0];
mt_dispch(i, rntStr[0]);
}
// Se for um ENTER, d?? a entrada na tela para mostrar
else if (chrAsHex == “d”)
{
// Captura e trata a entrada
if (dados[i] != “”)
{
// Faz o processamento do que desejar em dados[i]
}
// Limpa tela
ReiniciaEntrada(i);
}
// Se for um ESC, apaga tudo
else if (chrAsHex == “1b”)
{
// Limpa tela
ReiniciaEntrada(i);
}
// Se for backspace apaga ??ltimo caractere do display
else if (chrAsHex == “8”)
{
if (dados[i].Length > 0)
dados[i] = dados[i].Substring(0, dados[i].Length – 1);
mt_backspace(i);
}
// Se apertar * toca o som
else if (chrAsHex == “2a”)
{
// Toca um sonzinho… depois publico o c??digo de tocar som!
}
}
i++;
}
}
Feito! Rode sua aplica????o e digite algo no teclado do microterminal e ver?? que o que est?? sendo digitado est?? indo tanto para a aplica????o quanto para o display (no microterminal). A sa??da seria algo do tipo:
Se precisar de sa??da na tela crie um outro WinForm que ocupe toda tela (sua) e captura o que est?? no display do microterminal. Assim ?? mais f??cil! Ou fa??a do jeito que sua imagina????o mandar! O mais dif??cil j?? passou…
Para quem quiser capturar os IPs de cada microterminal, utilize os m??todos:
[DllImport(“pmtg.dll”, CallingConvention = CallingConvention.StdCall, SetLastError = true)]
public static extern TTABSOCK mt_connectlist();
ou
[DllImport(“pmtg.dll”, CallingConvention = CallingConvention.StdCall, SetLastError = true)]
public static extern StringBuilder mt_ipfromid(int id);
Nessas fun????es tem algumas particularidades muito chatinhas de se trabalhar (veja no manual das fun????es) por isso deixei de lado. Principalmente a classe:
[StructLayout(LayoutKind.Sequential)]
public struct TTABSOCK
{
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 255)]
public UInt32[] TSOCK;
}
Quem quiser se aventurar nela, fica a?? a dica. Esse artigo merecia at?? uma v??deo-aula, mas a pregui??a ??s vezes ?? muito forte! ?? at?? legalzinho ter esses desafios, mas ??s vezes se torna muito chato quando n??o consegue. Enfim, quanto menos se estressar, melhor! Ent??o tente usar as implementa????es que j?? a Gertec j?? disponibilizou, caso n??o queira aventurar, a menos que precise de algo mais espec??fico.
Falar que nem Marcoratti: “eu sei, ?? apenas ASP.NET, mas eu gosto!”