// Скрипт импорт из Протек (Саратов)

const UseBarcode = true; // определяет, будет ли использоваться штрихкод, поставляемый поставщиком

var
    Header :TStringList; // предопределена приложением
    Data :TStringList; // предопределена приложением
    ExportAgent :string; // все определены приложением
    FileName :string;

    ExportAgentName :string;
    ExportNakladNom :string;
    ExportNakladID_DOC :string;
    ExportNakladData :string;
    ExportNakladSklad :string;
    ExportSkladName :string;

const ListSeparator = ' от ';

var Table :TTable;
    num_doc :string;
    Item :variant;
    i,k :integer;
    OpenDialog :TOpenDialog;
    ThF_FIRSTPRICE,thf_SROK,thF_SrokSertif,ThF_NameSertif,thF_Amount,thF_NDS,
    thF_Name,thF_Seria,thF_Sertif,thF_NameProducer,thf_Tamog,
    thF_Country,thF_Regn,thF_Price,thF_BarCode,thF_DateSertif,thF_Em :TField;
    thF_Price_Reg :TField;
    str,str1 :string;
    Field :TField;

    InvoicesList :TStringList;

    List :TStringlist;
    List1 :tStringLIst;
    FirstI :integer;
    Strval :string;
    INtVal :integer;
    StrVal3 :string;
    StrVal1 :string;
    PriceWithNDS :currency;
    PriceWithoutNDS :currency;
    FloatVal2 :double;

    function GetItemValue(Item :Variant; Name :String) :string;
    begin
       try
         result := vartostr(Item.attributes.getNamedItem(Name).value);
       except
         result := ''
       end;
    end;

    function FindField(const Table :tDataset; const FieldNames :string; const FieldDescription :string) :TField;
    var List :TstringList;
        i :integer;
    begin
       result := nil;
       List := tstringlist.create;
       try
          stringtolist(FieldNames,List,';,');
          if list.count=0 then RaiseException('не указаны имена полей в функции FindField');

          for i := 0 to List.count-1 do
          begin
            Result := Table.findfield(List[i]);
            if result<>nil then break;
          end;
          if (Result=nil) and (FieldDescription>'') then RaiseException('не найдено поле "'+FieldDescription+'" ('+FieldNames+')');
       finally
          List.free;
       end;

    end;

    function IFSTR(BoolVal :boolean; const TrueVal :string; const FAlseVal :string = '') :string;
    begin
      if BoolVal then Result := TrueVal
      else Result := FalseVal;
    end;


    function GetValueFromListDialog(List :Tstrings) :string;
    var Form :TForm;
        ListBox :TListBox;
        Panel :TPanel;
        Btn :TButton;
    begin

      result := '';
      if List=nil then  exit;

      Form := TForm.create(nil);

      try
        ListBox := TListBox.create(Form);
        with Listbox do
        begin
          parent :=Form;
          Items.text := List.text;
          if List.count>0 then ItemIndex := 0;
          align :=alclient;
          Sorted := true;
        end;


        Form.caption := 'Выберите накладную:';
        form.position := poScreenCenter;
        form.borderstyle := bsDialog;
        form.width := 500;
        form.height := 400;
        Form.Font.Size := Form.Font.Size +4;

        Panel := tPanel.create(Form);
        with Panel do
        begin
          parent := Form;
          align := albottom;
          height := 35;
          BevelInner :=bvNone;
          BevelOuter := bvNone;

          (*
          with TBevel.create(Form) do
          begin
            Parent := panel;
            Align :=alTop;
            Shape := bsTopLine;
            Height := 4;
          end;
          *)

          Btn := TButton.create(Form);
          with Btn do
          begin
            Parent := Panel;
            Caption := 'Отмена';
            ModalResult := mrCancel;
            Cancel := true;
            width := 70;
            Top := 6;
            Left := Form.ClientWidth - Width - 20;
          end;


          with TButton.create(Form) do
          begin
            Parent := Panel;
            Caption := 'Ok';
            ModalResult := mrOk;
            Default := true;
            Width := Btn.width;
            Top := Btn.Top;
            Left := Btn.left- Width -10;
          end;

        end;

        if form.showmodal = mrok then
        begin
          if ListBox.Itemindex <0
          then CreateHintW('Вы не выбрали ни одной накладной')
          else Result := ListBox.Items[ListBox.Itemindex];
        end

      finally
        Form.free
      end;
    end;

 begin
     if filename = ''
     then begin
       OpenDialog := TOpenDialog.create(nil);
       try
         OpenDialog.Title := 'Укажите файл';
         OpenDialog.FileName  := AppIniFile.readstring('plugins_protek','filename','');
         opendialog.initialdir := extractfilepath(opendialog.filename);
         OpenDialog.Filter := 'Файлы-накладные(*.dbf,*.sst)|*.dbf;*.sst|Все файлы(*.*)|*.*';
         while  OpenDialog.execute do begin
           if not fileexists(OpenDialog.FileName) then CreateHint('Укажите правильно файл ')
           else begin
             Filename := OpenDialog.FileName;
             AppIniFile.writestring('plugins_protek','filename',OpenDialog.filename);
             break;
           end;
         end;
       finally
         OpenDialog.free;
       end;
     end;

     if FileName = '' then exit;

     //showmessage(uppercase(extractfileext(filename)));

     InvoicesList:= Tstringlist.create;

     with TMyWait.create('Идет импорт накладной') do
     try
       if uppercase(extractfileext(filename)) = '.DBF'
       then begin

         Table := TTable.Create(nil);
         try

          table.tabletype := ttFoxpro;

          Table.TableName := gettempdir + extractfilename(FileName);
          filedelete(Table.tablename);
          Table.tabletype := ttfoxpro;
          copyfile(FileName,Table.TableName,False);
          SetTableLangDriver(Table,'db866ru0');
          //CheckDBFCP(Table.TableName,38); //866DOS

          Table.Open;
          Table.First;

          if table.isempty then begin
             createhintw('Таблица пуста');
             exit;
          end;

          if table.findfield('nomerdoc')<>nil
          then begin

            table.first;
            while not table.eof do
            begin
              Str := Table.fieldbyname('nomerdoc').asstring + ListSeparator + Table.fieldbyname('datadoc').asstring;

              If InvoicesList.indexof(Str)<0
              then InvoicesList.add(Str);

              table.next;
            end;

            if InvoicesList.count>1
            then Str := GetValueFromListDialog(InvoicesList)
            else Str := INvoicesList[0];

            if Str='' then begin
              CreateHintW('Не выбрана накладная');
              exit;
            end;

            Header.Values[ExportNakladNom] := copy(str,1,pos(listSeparator,Str)-1);

            table.filter := '[nomerdoc]='+Quotedstr(Header.Values[ExportNakladNom]);
            Table.filtered := true;

            Header.Values[ExportNakladData] := copy(str,pos(listSeparator,Str)+length(ListSeparator),length(Str));

            CreateHint('Дата: ' + Header.Values[ExportNakladData] +', номер: '+ Header.Values[ExportNakladNom]);
          end
          else begin
            raiseexception('Таблица имеет формат, не согласованный с поставщиком. Обратитесь в техподдержку');
          end;


          data.Clear;

          {
          list.Add('srok'#9'barcode'#9'sroksertif'#9'sertifname'#9+
            'upa'#9'nn'#9'article'#9'kart'#9'kol'#9'em'#9'rosn'#9'nds'#9'ed'#9+
            'name'#9'firstcena'#9'seria'#9'sertif'#9'analize'#9'producer'#9+
            'producername'#9'nalog5'#9'idanalize'#9'analizename'#9'datasertif'#9+
            'dateanalize'#9'reestrcena'#9'tamog'#9'producercountry'#9+
            'producercountryname'#9'maxprocent'#9'passport'#9'datepassport'#9+
            'sorti'#9'gost'#9'regn'#9'summa'#9'cena'#9'itogcena'#9'itogsumma');
          }


          thF_FIRSTPRICE := FindField(Table, 'PRODCOST','Цена завода');
          thF_SROK :=  FindField(Table,'SERTDAT','Срок годности');
          thF_SROKSERTIF := FindField(Table,'SERTDAT','Срок действия сертификата');
          thF_NAMESERTIF := FindField(Table,'ORGAN','Кем выдан сертификат');
          thF_AMOUNT := FindField(Table,'KOL','Количество единиц');
          thF_NDS := FindField(Table,'PROCNDS','Ставка НДС');
          thF_NAME := FindField(Table,'NAME','Наименование товара');
          thF_Seria := FindField(Table,'SERIA','Серия');
          thF_Sertif := FindField(Table,'REGNOMER','Сертификат');
          thF_NameProducer := FindField(Table,'PRODUCT','Производитель');
          thF_tamog := FindField(Table,'GTD','Номер таможенной декларации');

          //DUE_DATE ? что за поле



          thf_country := findfield(table,'country;STRANA','Страна');

          thF_RegN := FindField(Table,'INP','Регистрационный номер');

          if UseBarcode
          then
            thF_Barcode := FindField(Table,'BARCODE','') // если нет, то не будет ошибки
          else
            thF_Barcode := nil;

          thF_DateSertif := findfield(table,'PASDAT','Дата выдачи сертификата');
          thF_Em := findfield(table,'package','');
          thF_Price_reg := findfield(table,'REESTR','');

          thF_Price := FindField(table,'SUMMA','Цена');


          data.Add('srok'#9'sroksertif'#9'sertifname'#9+
            'kol'#9'nds'#9+
            'name'#9+
            'firstcena'#9 + //ifstr(table.findfield('num_nak')<>nil,'firstcena','firstcenaplusnds')+#9+
            'seria'#9'sertif'#9+
            'producername'#9+
            'tamog'#9+
            ifstr(thf_Country<>nil,'countryname'+#9,'')+
            'regn'#9+
            'SUMMA'#9+ //ifstr(table.findfield('num_nak')<>nil,'cenanotnds','cena')+#9

            ifstr(thf_barcode<>nil,'barcode'+#9,'')+
            ifstr(thf_datesertif<>nil,'datesertif'+#9,'')+
            ifstr(thf_em<>nil,'em'+#9,'')+
            ifstr(thf_price_reg<>nil,'price_reg'+#9,'')
            );

          table.first;

          while not Table.eof do begin
            Str := '';

            str :=
               thF_Srok.asstring+#9+
               thf_sroksertif.asstring+#9+
               thf_namesertif.asstring+#9+
               thf_amount.asstring+#9+
               thf_NDS.asstring +#9+
               thf_name.asstring+#9+

               thF_FIRSTPRICE.asstring+#9+
               thf_SEria.asstring+#9+
               thf_SErtif.asstring+#9+
               thf_NameProducer.asstring+#9+
               thf_Tamog.AsString+#9;

            if thF_Country <> nil
            then
               Str := Str + thF_Country.AsString+#9;


            Str := Str +
               thf_regn.asstring+#9+
               thF_Price.asstring +#9;



            if  thF_Barcode<>nil
            then str:= str + thF_Barcode.asstring+#9;

            if  thf_datesertif<>nil
            then str := str + thF_DateSErtif.asstring+#9;

            if  thf_em<>nil
            then str := str + thF_em.asstring+#9;

            if  thf_price_reg<>nil
            then str := str + thF_price_reg.asstring+#9;

            data.Add(str);
            Table.Next;
          end;
         finally
            Table.free;
         end;
       end
       else
       if uppercase(extractfileext(filename)) = '.SST'
       then begin

         List := TStringList.create;
         List1 := TsTringLIst.create;
         try

          List.LoadFromFile(FileName);

          if List.count<2 then begin
             raiseexception('ФАйл накладной имеет менее 2 строк. Прием невозможен. Обратитесь в техподдержку');
          end;


          //list.casesensitive := false;

          FirstI :=  list.IndexOf('[Header]');
          if FirstI <0
          then begin
             raiseexception('Неправильный формат файла - не найден раздел [Header]');
          end;

          inc(FirstI);
          while (List.count>FirstI) and (List[FirstI] ='-') and (uppercase(list[FirstI]) <>'[BODY]')
          do Inc(FirstI);

          if (List.count>FirstI) and (uppercase(list[FirstI])<>'[BODY]')
          then begin
            StrVal := List[FirstI];

            if StrVal >''
            then begin
              IntVal := pos(';',StrVal);
              if IntVal >-1
              then begin
                Header.add(ExportNakladNom+'='+copy(strval,1,IntVal-1));
                StrVal := copy(StrVal,IntVal+1,100);
                IntVal := pos(';',strval);
                if IntVal>0
                then strVal := copy(StrVal,1,IntVal-1);

                strVal := Copy(strval,1,10);  // заглушка от времени

                if StrVal >''
                then Header.add(ExportNakladData+'='+formatdatetime('dd.mm.yyyy',strtodateprotected( StrVal)));


              end
            end
            else begin
               bvMessageError('Неправильный формат файла - пестой раздел [Header]');
               exit;
            end;


          end
          else begin
             bvMessageError('Неправильный формат файла - не найдено содержимое раздела [Header]');
             exit;
          end;


          FirstI := list.indexof('[BODY]');
          if FirstI <0
          then begin
             bvMessageError('Неправильный формат файла - не найден раздел [Body]');
             exit;
          end;


          for i := 0 to FirstI-1 do list.Delete(0);

          list[0] :='name'#9'producername'#9'countryname'#9+
                    'kol'#9'FirstCena'#9'cenanotnds'#9+
                    'GTD'#9+
                    'sertif'#9+
                       'datesertif'#9+
                       'sertifname'#9+
                    'seria'#9'srok'#9+

                    'barcode'#9 +
                    //'datereestr'#9+
                    'price_reg'+#9+

                    'nds';

          //list.text := StringReplace(list.text,';',#9,[rfReplaceAll]); //почему-то AV здесь


          for i := 1 to list.count-1 do begin

            strval3 := list[i];

            stringtolist(StrVAl3,List1,';');


            StrVAl1 := List1[1];                              // name
            StrvAl1 := StrVAl1 + #9 + list1[2];              // producername


            StrvAl1 := StrVAl1 + #9 + List1[3];             // countryname

            StrvAl1 := StrVAl1 + #9 + List1[4];             // kol

            PriceWithNDS := StrToFloatProtected( List1[5]); // цена с НДС

            FloatVal2 := strtofloatprotected(List1[6]);
            if FloatVal2 <>0
            then
              StrvAl1 := StrVAl1 + #9 + List1[6]             // firstcena
            else
              StrvAl1 := StrVAl1 + #9 + List1[7];             // FirstCena, если была пустая

            PriceWithoutNDS := strtofloatprotected(List1[7]);

            StrvAl1 := StrVAl1 + #9 + List1[7];             // cenanotnds

            StrvAl1 := StrVAl1 + #9 + List1[11];             // ГТД

            /////////////////////////////

              StrvAl := List1[12];

               //43AO55^POCC DE.ФM02.A71359(P-318-2716-4)^01.06.07  МЦС г.Москва
               //010304^POCC.DE.ФM02.A87770^05.10.2004 ОАО "МЦС" Москва010304^140059^28.05.2005
               //0505^77.99.11.3.У.2840.9.04^16.09.2004 ФЗ ППиБ
               //103103103BB^004677.Р.643.03.2003^17.03.2003 МЗРФ103103103BB^77.99.20.919.Б.000187.02.04^13.02.2004 ДГСЭН

              IntVal := pos('^',StrVal); // пропускаем первую галочку - серия
              delete(strval,1,intval);

              IntVal := pos('^',StrVal);
              strval1 := strval1+ #9+ copy(strval,1,intval-1); // сертификат
              delete(strval,1,intval);


              IntVal := pos(' ',StrVal); // дата выдачи сертификата
              strval1 := strval1+ #9+ copy(strval,1,intval-1);
              delete(strval,1,intval);

              intval := Pos('',strval);
              if intval >0
              then Delete(strval,intval,1000);

              strval1 := strval1+ #9+ strval; // сертифицирующий орган


            /////////////////////////////

            StrvAl1 := StrVAl1 + #9 + List1[13];             // СЕРИЯ


            StrvAl1 := StrVAl1 + #9 + List1[15]; // Срок годности (дата)
            StrvAl1 := StrVAl1 + #9 + List1[16];             // barcode
            //StrvAl1 := StrVAl1 + #9 + List1[14]; // Дата выпуска препарата


            (*
            StrvAl3 := Copy(StrVal3,FirstI+Length(strval)+1,1000);
            StrvAl := GetWord(StrVAl3,FirstI,';',False);
            while (Length(strval)>0) and  (StrVal[1]='0') do
               Delete(StrVal,1,1);

            if not UseBarcode
            then STrvAl := ''
            else
            if Length(strVAl)>12 then Delete(StrVal,13,1000);
            *)


           // StrvAl1 := StrVAl1 + #9 + List1[17];            // datereestr


            StrvAl1 := StrVAl1 + #9 + List1[18];           // цена реестра

            //StrvAl3 := Copy(StrVal3,FirstI+Length(strval)+1,1000);
            //StrvAl := GetWord(StrVAl3,FirstI,';',False);

            //createhint(floattostr(pricewithnds)+':'+floattostr(pricewithoutnds)+':'+inttostr(round((PriceWithNDS / PriceWithoutNDS - 1) * 100)));

            if PriceWithoutNDS = 0    // Ставка НДС
            then
              StrVAl1 := strVAl1+#9#9
            else
            try
              strval1 := strval1 +#9 +inttostr(round((PriceWithNDS / PriceWithoutNDS - 1) * 100))+#9;
            except
            end;

            list[i] := StrVal1;
          end;

          data.text := list.text;



         finally
            List1.free;
            List.free;
         end;
       end
       else
            raiseexception('Таблица имеет формат, не согласованный с поставщиком. Обратитесь в техподдержку');

     finally
       Invoiceslist.free;
       free
     end;

     createhint('Операция завершена');
 end.
