webix画面からQRラベル印刷機能の実装
前回、備品管理用機能を実装した事例を紹介しましたが、その機能で、保管場所一覧にチェックボックスを追加して、複数の保管場所情報を選択して、QRコードラベルを印刷機能を紹介します。印刷したQRコードラベルは、それぞれの保管場所に貼ります。備品のQRコードと保管場所のQRコードを使用して、備品の保管先をQRコード操作で対応づける機能が最終目標です。
今回は、保管先のQRコードラベル印刷を紹介します。今後、備品についても保管先と同じように機能を実装(一覧、新規登録、編集、削除、EXCELインポート、QRコード印刷など)すればQRコードをスマホで操作し、備品管理ができるようになります。
前回から修正した機能は、以下のとおりです。
webix01/view/LC0010/LC0010_locations_lists.php 一覧にチェックボックスと印刷ボタン実装(変更)
webix01/rest_api/LC0010/LC0019_print_locationlabels.php QRコードラベル印刷指示用REST_API(追加)
webix01/config/app01.php 各種環境定義用config(変更)
印刷機能(C#) tepra_print.exe QRコードに名称を含めないフォーマットに変更
印刷フォーム bihin_12.tpe 項目情報の幅変更(全長を短くするため)
webixのチェックボックス機能を使い、一覧から複数行を選択できるようにしました。選択した状態でQR印刷をクリックすると、サーバ側(PHP)に印刷要求をだします。LC0019_print_locationlabels.phpで複数行の情報を受け取り、順番にQRラベル印刷指示をcurlコマンドを使って、TEPRA印刷機能(tepra_print.exe)が動作しているPCに依頼します。
tepra_print.exeでは、要求を受信すると、CSVファイルを作成してTEPRAプリンタに印刷指示を出します。
一覧画面では、チェックした行の情報を取り出し、連想配列に代入後に、JSON.stringifyで、文字列化してサーバに送信します。
サーバ側(PHP)では、json_decodeを使って、連想配列に戻し、1行づつ、印刷指示をcurlで依頼します。PHPが動作スルサーバ(Linux)からTEPRA印刷が動作するWindows10のPCへの依頼は、PC側のC#で簡易Webサーバを実装して要求を受信し、PHPからの印刷要求の情報をCSVにしてTEPRAプリンタに印刷指示を出します。
印刷が正常に終了すると、該当レコードのメモに、印刷済と追記してデータを更新します。
以下は、5件選択して印刷をした結果画面です。メモに印刷済と追記されます。
実際に印刷したラベルです。5枚印刷されました。
画面ソースの修正箇所だけ記載します。LC0010_locations_lists.php
//QRラベル印刷処理
function LC0010_print_QRlabel(print_lists){
var formData = new FormData();
var access_key = Get_AccessKey();
formData.append("accesskey",access_key);
formData.append("userid", my_local_session.userid);
formData.append("job_info","LC0010");
formData.append("print_lists",JSON.stringify(print_lists));
var xhr =webix.ajax().sync().post("<?php echo SUB_FOLDER; ?>/rest_api/LC0010/LC0019_print_locationlabels.php",formData);
var resp = JSON.parse(xhr.responseText);
if(resp.resp =="ok"){
LC0010_select_locations_lists();
webix.message({type:"success",text:"QRラベルを印刷しました。"});
return;
}
else{
webix.alert("QRラベルの印刷処理でエラーしました。 code="+resp.error_code);
return;
}
}
{ align:"top", body:{ view:"button",id:"LC0010_print_QRlabel_btn", name:"LC0010_print_QRlabel_btn", value: "QR印刷", align:"center", width: 110, css:"webix_danger",
click:function(){
var print_lists = [];
var cnt = 0;
$$("LC0010_table").eachRow(function(row){
const record = $$("LC0010_table").getItem(row);
if(record.ch1 == 1){
cnt += 1;
print_lists.push(record);
}
});
if(cnt == 0){
webix.alert("QRラベル印刷<br>対象の行がありません。");
return;
}
else{
webix.confirm({
title:"確認",
ok:"はい",
cancel:"いいえ",
text:cnt+"件、<br>QRラベルを印刷しますか?"
})
.then(function(result){
LC0010_print_QRlabel(print_lists);
})
.fail(function(){
webix.message({type:"debug",text:"印刷操作をキャンセルしました。"});
});
}
}
}},
var grid1 = {
id: "LC0010_table",
name: "LC0010_table",
type:"clean",
view:"datatable",
width:800,
resizeColumn:true,
resizeRow: { headerOnly:true },
columns:[
{ id:"ch1", header:{ content:"masterCheckbox", contentId:"cm1"}, template:"{common.checkbox()}", width:40 },
{ id:"id",header:["id",{content:"textFilter"}],width:100,sort:"int", css:{"text-align":"right"}},
{ id:"code",header:["保管先コード",{content:"textFilter"}],width:100,sort:"int", css:{"text-align":"right"}},
{ id:"area",header:["場所",{content:"selectFilter"}],width:120,sort:"int", css:{"text-align":"right"}},
{ id:"shelf",header:["棚",{content:"selectFilter"}],width:80,sort:"int", css:{"text-align":"right"}},
{ id:"place",header:["位置",{content:"selectFilter"}],width:80,sort:"int", css:{"text-align":"right"}},
{ id:"name",header:["保管先名称",{content:"textFilter"}],width:250},
{ id:"memo",header:["メモ",{content:"textFilter"}],width:200},
{ id:"editmode",header:"editmode",width:10,hidden:true} //1:閲覧 2:更新 3:新規 4:削除
],
新規に追加したLC0019_print_locationlabels.phpです。
<?php
//======================================================================
//File Name : LC0019_print_locationlabels.php
//Encoding : UTF-8
//Creation Date : 2024-03-29
//
//Copyright © 2024 sunsunfarm. All rights reserved.
//
//This source code or any portion thereof must not be
//reproduced or used in any manner whatsoever.
//======================================================================
header("Content-Type: text/javascript; charset=utf-8");
$myfilename = basename(__FILE__); //自分自身のファイル名取得
$userid = '';
$logheader = 'userid='.$userid.', '.$myfilename.':';//ログ出力時のヘッダー情報(自ファイル名,ログインIDを付与)
if($_SERVER["REQUEST_METHOD"] != "POST"){
error_log($logheader.' not POST method');
header("HTTP/1.0 404 Not Found");
return;
}
//指定パラメータでUPDATEする
//POST情報からパラメータ取得
$accesskey = 0;
if(isset($_POST['accesskey'])){
if(is_numeric($_POST['accesskey'])){
$accesskey = $_POST['accesskey'];
$error_flag = 1;
}
}
else{
$error_flag = -1;
error_log($logheader.' accesskey not found');
header("HTTP/1.0 404 Not Found");
exit;
}
if(isset($_POST['userid'])){
$userid = $_POST['userid'];
$error_flag = 1;
}
else{
$error_flag = -1;
error_log($logheader.' userid not found');
header("HTTP/1.0 404 Not Found");
exit;
}
$job_info = "LC0019";
if(isset($_POST['job_info'])){
$job_info = $_POST['job_info'];
}
$logheader = 'userid='.$userid.', '.$myfilename.':';//ログ出力時のヘッダー情報(自ファイル名,ログインIDを付与)
include('../../commonlib/svr_common_lib_v2.php'); //
$config_obj = get_config_obj();
$print_mode = $config_obj::get('app01.print_mode'); //QRコード印刷モード blank:通常 test:logのみ
$tepra_print_serve_ip = $config_obj::get('app01.tepra_print_server'); //QRコード印刷指示PCのIPアドレス
$tepra_print_serve_port= $config_obj::get('app01.tepra_print_serve_port'); //QRコード印刷指示URLのポート番号(8081)
//アクセスキーチェック
if(Chk_AccessKey($accesskey)){
}
else{
error_log($logheader.' accesskey check error');
header("HTTP/1.0 404 Not Found");
exit;
}
$print_lists = array();
if(isset($_POST['print_lists'])){
$print_lists_str = $_POST['print_lists'];
$print_lists = json_decode($print_lists_str ,true);
}
/**
* curl でGET通信を行う
*
* @param [string] $url - 'http://example.com/'
* @return [string|boolean] - 成功: 取得結果, 失敗: false
*/
function sendGetRequest($url){
$option = [
CURLOPT_RETURNTRANSFER => true, // 文字列として返す
CURLOPT_TIMEOUT => 9000, // タイムアウト時間
CURLOPT_SSL_VERIFYPEER => false, // サーバ証明書の検証をしない
];
// curl初期化
$ch = curl_init($url);
curl_setopt_array($ch, $option);
$body = curl_exec($ch);
$info = curl_getinfo($ch);
$errorNo = curl_errno($ch);
$errorMsg = curl_error($ch);
// 「CURLE_OK」以外はエラーなのでエラー情報を返す
if ($errorNo !== CURLE_OK) {
// 詳しくエラーハンドリングしたい場合はerrorNoで確認
// タイムアウトの場合はCURLE_OPERATION_TIMEDOUT
return $errorNo . " : " . $errorMsg;
}
// 200以外のステータスコードは失敗なのでそのステータスコードを返す
if ($info['http_code'] !== 200) {
return $info['http_code'];
}
$responseArray = json_decode($body, true); // JSON を配列に変換
return($responseArray);
}
//
//メインルーチン
//
$config_obj = get_config_obj();
//データベース接続する(MariaDB)
$dbh = mariadb_connect($config_obj,'app01','webix-server_redmine001');
if($dbh == false){
//エラー応答
$resp = "ng";
$error_code = -2;
$json_data = json_encode(compact("resp","error_code"),JSON_UNESCAPED_UNICODE);
echo $json_data;
exit;
}
try{
$update_sql1 = "UPDATE locations SET memo = ?,updated_userid = ?,updated_on= ? where id= ?";
$update_stmt = $dbh->prepare($update_sql1);
// 静的プレースホルダを指定
$dbh->setAttribute(PDO::ATTR_EMULATE_PREPARES, false);
// エラー発生時に例外を投げる
$dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
//トランザクション処理を開始
$dbh->beginTransaction();
$resp = "ok";
$error_code = 0;
for($i=0;$i<count($print_lists);$i++){
//制御先のURLとパラメータ準備
if($print_mode == ""){
$url = 'http://'.$tepra_print_serve_ip.':'.$tepra_print_serve_port.'/?code='.$print_lists[$i]["code"].'&label='.urlencode($print_lists[$i]["name"]);
}
else{
$url = 'http://'.$tepra_print_serve_ip.':'.$tepra_print_serve_port.'/?code='.$print_lists[$i]["code"].'&label='.urlencode($print_lists[$i]["name"]).'&mode='.$print_mode;
}
error_log($logheader.' url=:'.$url);
//ラベル印刷
$rs = sendGetRequest($url);
if($rs != 0){
error_log($logheader.' sendGetRequest error rs='.$rs);
$resp = "ng";
$error_code = -10;
$dbh = null;
$json_data = json_encode(compact("resp","error_code"),JSON_UNESCAPED_UNICODE);
echo $json_data;
exit;
}
else{
$memo = $print_lists[$i]["memo"];
if($memo == ""){
$memo = "印刷済";
}
else{
$memo = $memo." 印刷済";
}
$update_stmt->bindValue(1,$memo);
$update_stmt->bindValue(2,$userid); //updated_userid
$update_stmt->bindValue(3,date("Y-m-d H:i:s"));
$update_stmt->bindValue(4,intval($print_lists[$i]["id"]),PDO::PARAM_INT);
$update_stmt->execute();
error_log($logheader.' update locations id=:'.$print_lists[$i]["id"]);
}
}
$dbh->commit();
$dbh = null;
error_log($logheader.' commit');
} catch(PDOException $e){
error_log($logheader.' 捕捉した例外: '.$e->getMessage()); //例外発生時の処理(エラー情報をログに格納してエラー応答)
$resp = "ng";
$error_code = -1;
}
$dbh = null;
$json_data = json_encode(compact("resp","error_code"),JSON_UNESCAPED_UNICODE);
//error_log($logheader.' $json_data ='.$json_data);
echo $json_data;
?>