webdata
DX推進をサポートする技術者向け情報提供サイト

初心者向けPHP・データベース入門

TOP >初心者向けPHP・データベース入門 >5.2配列を使った月毎売上集計一覧の作成

配列を使った月毎売上集計一覧の作成

 2024-06-16 (更新日:2024-07-20)

<学習する内容>

 配列を使って4章の月毎の売上集計を作成しながら理解します。
配列に慣れるまでは混乱するかもしれませんが、配列の中身をprint_r()関数を使って繰り返し、 一つひとつ確認しながら理解を進めて下さい。理解には数をこなして慣れるしかありません。

3)売上一覧の作成

下記のサンプルプログラムをコピーもしくはダウンロードし指定のフォルダーに配置してください。
ファイル名:saleslist5_2.php
配置先:c:\xampp\htdocs\
配置先URL:http://localhost/saleslist5_2.php
サンプルプログラム名:saleslist5_2.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
<?php
 
//DBへの接続
include('/xampp/data/conn.php');
 
//POSTの受信
$fnc = filter_input(INPUT_POST, 'fnc');
$select_id = filter_input(INPUT_POST, 'select_id');
$select_date = filter_input(INPUT_POST, 'select_date');
$select_sectionCode = filter_input(INPUT_POST, 'select_sectionCode');
$select_staffCode = filter_input(INPUT_POST, 'select_staffCode');
 
//年度option
$day = date("Y-m-01");                  //当月1日
$day = '2024-04-01';                    //デモ用のデータ(2024年度)を表示のため記述
$num = 10;                              //表示期数
$opt_date = null;                       //初期化
for ($i = 0;$i <= $num;$i++) {
    $s = $i * 6;                        //半期6か月毎
    $FY = date("Y",strtotime("-$s month",strtotime($day)));
    $month = date("n",strtotime("-$s month",strtotime($day)));
    $date = date("Y-m-d",strtotime("-$s month",strtotime($day)));
    if ($month >= 4 and $month < 10){
        $opt_date .= '<option value="'.$date.'" >'.$FY.'上期';
    } else {
        $opt_date .= '<option value="'.$date.'" >'.$FY.'下期';
    }
}
//年度表示と検索用データ生成
if (empty($select_date)){
    $select_date = $day;
}
$selectFY = date("Y",strtotime($select_date));
$n = date("n",strtotime($select_date));         //1桁月
if ($n >= 4 and $n < 10) {
    $startDate = $selectFY.'-04-01';
    $endDate = $selectFY.'-10-01';
    $term = '上期';
    $select_date_display = $selectFY.$term;
} else {
    $selectFY1 = $selectFY + 1;
    $startDate = $selectFY.'-10-01';
    $endDate = $selectFY1.'-04-01';
    $term = '下期';
    $select_date_display = $selectFY.$term;
}
for ($i = 0;$i <= 6;$i++) {
    $ARY_MONTH[$i] = date("Ym",strtotime("+$i month",strtotime($select_date)));
    $ARY_MONTH_INDEX[$i] = date("Y年n月",strtotime("+$i month",strtotime($select_date)));
    if ($i == 6) {              //半期計値用
        $ARY_MONTH[$i] = 'sum';
        $ARY_MONTH_INDEX[$i] = $term.'計';
    }
}
 
//部署option
$sql = "SELECT sectionCode,sectionName FROM Msection;";
$res = mysqli_query($conn,$sql) or die("error $sql");
    $OPTION_SECTION = '<option style="color:#2E2E2E;" value="">解除';
while($row = mysqli_fetch_array($res)){
    $sectionCode_opt = $row["sectionCode"];
    $sectionName_opt = $row["sectionName"];
    $OPTION_SECTION .= '<option style="color:#2E2E2E;" value="'.$sectionCode_opt.'">'.$sectionName_opt;
}
//部署option
$sql = "SELECT sectionCode,sectionName FROM Msection;";
$res = mysqli_query($conn,$sql) or die("error $sql");
    $OPTION_SECTION = '<option style="color:#2E2E2E;" value="">解除';
while($row = mysqli_fetch_array($res)){
    $sectionCode_opt = $row["sectionCode"];
    $sectionName_opt = $row["sectionName"];
    $OPTION_SECTION .= '</option><option style="color:#2E2E2E;" value="'.$sectionCode_opt.'">'.$sectionName_opt;
}
//部署検索
if(empty($select_sectionCode)){
    $select_sectionName = '部署検索';
    $where_and_section = null;                  //初期化
}else{
    $where_and_section = "and a.sectionCode = $select_sectionCode";
    $sql = "SELECT sectionCode,sectionName FROM Msection where sectionCode = $select_sectionCode;";
    $res = mysqli_query($conn_demo,$sql) or die("error $sql");
    $row = mysqli_fetch_array($res);
        $select_sectionName = $row["sectionName"];
}
 
//担当option
$OPTION_STAFF = null;                   //初期化
if(strlen($select_sectionCode)){
    $sql = "SELECT staffCode,staffName FROM Mstaff where sectionCode = $select_sectionCode;";
    $res = mysqli_query($conn_demo,$sql) or die("error $sql");
        $OPTION_STAFF = '</option><option style="color:#2E2E2E;" value="">解除';
    while($row = mysqli_fetch_array($res)){
        $staffCode_opt = $row["staffCode"];
        $staffName_opt = $row["staffName"];
        $OPTION_STAFF .= '</option><option style="color:#2E2E2E;" value="'.$staffCode_opt.'">'.$staffName_opt;
    }
}
//担当者名検索
if(empty($select_staffCode)){
    $select_staffName = '担当検索';
    $where_and_staff = null;                    //初期化
}else{
    $where_and_staff = "and a.staffCode = $select_staffCode";
    $sql = "SELECT staffCode,staffName FROM Mstaff where staffCode = $select_staffCode;";
    $res = mysqli_query($conn_demo,$sql) or die("error $sql");
    $row = mysqli_fetch_array($res);
        $select_staffName = $row["staffName"];
}
 
//$_POSTデータの保持
$OPT_DATA = null;                   //初期化
$OPT_DATA .= '<input type="hidden" name="select_sectionCode" value="'.$select_sectionCode.'">';
$OPT_DATA .= '<input type="hidden" name="select_staffCode" value="'.$select_staffCode.'">';
$OPT_DATA .= '<input type="hidden" name="select_date" value="'.$select_date.'">';
 
//HTMLテーブル項目(年月)代入
$HTML = null;                   //初期化
$HTML .= '<tr bgcolor="#D3D3D3" style="height:24px;" align="center">';
$HTML .= '  <td width="100" rowspan="2">部署</td>';
$HTML .= '  <td width="90" rowspan="2">担当者</td>';
for ($i = 0;$i < count($ARY_MONTH); $i++) {
    $HTML .= '<td width="160" colspan="2">'.$ARY_MONTH_INDEX[$i].'</td>';
}
$HTML .= '</tr>';
 
//HTMLテーブル項目(売上・利益)代入
$HTML .= '<tr bgcolor="#D3D3D3" style="height:24px;" align="center">';
for ($i = 0;$i < count($ARY_MONTH); $i++) {
    $HTML .= '<td>売上</td>';
    $HTML .= '<td>利益</td>';
}
$HTML .= '</tr>';
 
//データ抽出&HTML代入
$sql = "SELECT sum(sales) AS sales
                    ,sum(profit) AS profit
                    ,staffCode
                    ,EXTRACT(YEAR_MONTH FROM salesDate) AS salesMonth
        FROM saleslist
        WHERE salesDate >= '$startDate'
        AND salesDate < '$endDate'
        GROUP BY staffCode,salesMonth
        ;";
$res = mysqli_query($conn,$sql) or die("er SELECT $sql");
while($row = mysqli_fetch_array($res)){
    $staffCode = $row["staffCode"];
    $month = $row["salesMonth"];
    $sales[$staffCode][$month] = $row["sales"];
    $profit[$staffCode][$month] = $row["profit"];
    $DETAIL[$staffCode][$month] = "<a onclick="\"window.open('sales_detail.php?select_staffCode=".$staffCode."&select_salesMonth=".$month."'," 'detail','width="800," height="300," menubar="no," toolbar="no," scrollbars="yes');\"">";
    //半期計集計
    if (isset($sales[$staffCode]['sum'])) {
        $sales[$staffCode]['sum'] += $row["sales"];
        $profit[$staffCode]['sum'] += $row["profit"];
    }
        $sales[$staffCode]['sum'] = $row["sales"];
        $profit[$staffCode]['sum'] = $row["profit"];
}
 
//担当データよりHTML文の生成
$sql = "SELECT a.staffCode,a.staffName,b.sectionName FROM Mstaff as a
            LEFT JOIN Msection AS b ON a.sectionCode = b.sectionCode
            WHERE a.staffCode > 0
            $where_and_section
            $where_and_staff
            ORDER BY a.sectionCode,a.staffCode;";
$res_staff = mysqli_query($conn,$sql) or die("er SELECT $sql");
while($row_staff = mysqli_fetch_array($res_staff)){
    $staffCode = $row_staff["staffCode"];
    $staffName = $row_staff["staffName"];
    $sectionName = $row_staff["sectionName"];
 
    $HTML .= '<tr bgcolor="white" style="height:24px;" align="right">';
    $HTML .= '  <td align="center">'.$sectionName.'</td>';
    $HTML .= '  <td align="center">'.$staffName.'</td>';
    for ($i = 0;$i < count($ARY_MONTH);$i++) {
        if ($i == 6) {
            $ARY_MONTH[$i] = 'sum';
        }
        //数値フォーマット
        if (isset($sales[$staffCode][$ARY_MONTH[$i]])) {
            $sales[$staffCode][$ARY_MONTH[$i]] = number_format($sales[$staffCode][$ARY_MONTH[$i]]/1000);
            $detail = null;                 //初期化
            if (isset($DETAIL[$staffCode][$ARY_MONTH[$i]])) {
                $detail = $DETAIL[$staffCode][$ARY_MONTH[$i]];
            }
            $HTML .= '  <td>'.$detail.$sales[$staffCode][$ARY_MONTH[$i]].'</a></td>';
        } else {
            $HTML .= '   <td></td>';
        }
        if (isset($profit[$staffCode][$ARY_MONTH[$i]])) {
            $profit[$staffCode][$ARY_MONTH[$i]] = number_format($profit[$staffCode][$ARY_MONTH[$i]]/1000);
            $HTML .= '  <td>'.$profit[$staffCode][$ARY_MONTH[$i]].'</td>';
        } else {
            $HTML .= '   <td></td>';
        }
    }
    $HTML .= '</tr>';
}
 
//HTML本文
echo <<<EOT
<!DOCTYPE html>
<html lang="ja">
 
<head>
<meta charset="utf-8">
<link rel="stylesheet" href="/style.css" type="text/css">
<style>
a:hover {
    color: red;
    font-weight:bold;
}
</style>
<title>売上一覧</title>
</head>
 
<body>
売上一覧
<table  style="font-size:12px;">
    <tr style="height:24px;" align="center">
    <form method="POST" action="{$_SERVER['PHP_SELF']}">
            {$OPT_DATA}
        <td width="70">
            <SELECT name="select_date" onchange="submit(this.form)" style="width:100px;height:26px;">
                <option value="{$select_date}" selected>{$select_date_display}
                {$opt_date}
            </select>
        </td>
    </form>
    <form method="POST" action="{$_SERVER['PHP_SELF']}">
            {$OPT_DATA}
            <input type="hidden" name="select_staffCode" value="">
        <td width="70">
            <SELECT name="select_sectionCode" onchange="submit(this.form)" style="width:100px;height:26px;{$clr_section}">
                <option value="{$select_sectionCode}" selected>{$select_sectionName}
                {$OPTION_SECTION}
            </select>
        </td>
    </form>
    <form method="POST" action="{$_SERVER['PHP_SELF']}">
            {$OPT_DATA}
        <td width="70">
            <SELECT name="select_staffCode" onchange="submit(this.form)" style="width:100px;height:26px;{$clr_staff}">
                <option value="{$select_staffCode}" selected>{$select_staffName}
                {$OPTION_STAFF}
            </select>
        </td>
    </form>
            <td width="800"></td>
            <td>単位:千円</td>
    </tr>
</table>
<table bgcolor="#a9a9a9" cellspacing="1px" style="font-size:12px;" >
{$HTML}
</table>
<br>
 
</body>
</html>
EOT;
 
?>
</a></option>
<解説>
 プログラムの構成は、検索データの受信やselect文のoption生成、検索実行です。
111行目からはHTML表示用データ抽出やHTML文生成、173行目からはHTML文を記述しております。
4章のsaleslist.phpに比べプログラムの行数が320行→236行に減りプログラムの読みやすくなっている(配列に慣れれば)と思います。

47行目for文 for文により利用する年月データ、項目表示用年月データを生成しております。
for文の$i <= 6 の6は6か月と半期計表示として6としております。0から始まるので6を半期としてます。
$ARY_MONTH これは年月毎の売上データ識別用として実績のキーとして利用します。
$ARY_MONTH_INDEX これはテーブル年月表示用として利用します。

121行目for文
$i < count($ARY_MONTH); count($ARY_MONTH)が7要素なので0~6で回している、

142行目GROUP BY srtaffCode,salesMonth
月別担当者毎に合計値を抽出

148行目$sales[$staffCode][$month]
月別担当者毎の合計値を担当者コードと売上年月をキーとして配列$salesに代入
同様に149行目利益と150行目明細ポップアップも代入
176行目for文
161行目のSQL文により担当者コードを抽出。この担当者コードとfor文による順番(年月)キーとしたデータを取出し $HTML文に代入します。

255行$HTMLの表示
$HTMLに代入されたHTMLを表示させます。
※この$HTMLがどう代入されて表示されるか、ブラウザにてソースを表示させて確認しながら学習してください。