回答編集履歴
2
修正
test
CHANGED
@@ -18,48 +18,266 @@
|
|
18
18
|
|
19
19
|
|
20
20
|
|
21
|
+
------
|
22
|
+
|
23
|
+
|
24
|
+
|
25
|
+
**追記**
|
26
|
+
|
27
|
+
|
28
|
+
|
29
|
+
※修正前の回答コードではシートIDの存在確認が正常にできていなかったため、
|
30
|
+
|
31
|
+
冒頭に`existSheetId(spreadsheet, sheetId)`関数を追加しております。
|
32
|
+
|
33
|
+
|
34
|
+
|
35
|
+
あと、動作確認の際にいくつか不具合を見つけたので修正しています。
|
36
|
+
|
37
|
+
赤色[行頭がマイナス記号]の行 -> 削除
|
38
|
+
|
39
|
+
緑色[行頭がプラス記号]の行 -> 追加
|
40
|
+
|
41
|
+
|
42
|
+
|
43
|
+
mergePdfs関数については、コメントに記載の通り、https://qiita.com/mat_aaa/items/d77320769b5ac837a98b に記載のコードを参照してください。
|
44
|
+
|
45
|
+
(長すぎて回答欄に書けないため)
|
46
|
+
|
47
|
+
|
48
|
+
|
49
|
+
|
50
|
+
|
21
51
|
```diff
|
22
52
|
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
//
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
+ i
|
38
|
-
|
39
|
-
+
|
40
|
-
|
41
|
-
+
|
42
|
-
|
43
|
-
+
|
44
|
-
|
45
|
-
+
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
var
|
54
|
-
|
55
|
-
var pd
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
(
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
53
|
+
|
54
|
+
|
55
|
+
// シートIDの存在を確認する関数。
|
56
|
+
|
57
|
+
// 指定したspreadsheetにsheetIdで指定したIDのシートが存在するか確認する。
|
58
|
+
|
59
|
+
// 引数:spreadsheet:対象のスプレッドシート。sheetId:確認対象のシートID
|
60
|
+
|
61
|
+
// 返値:存在する場合true, 存在しない場合false。
|
62
|
+
|
63
|
+
+function existSheetId(spreadsheet, sheetId){
|
64
|
+
|
65
|
+
+ var sheets = spreadsheet.getSheets();
|
66
|
+
|
67
|
+
+ for(var i = 0; i < sheets.length; i++){
|
68
|
+
|
69
|
+
+ if(sheets[i].getSheetId() === sheetId) return true; // 「===」にしないと、sheetIdが空白の場合trueとなってしまう。
|
70
|
+
|
71
|
+
+ }
|
72
|
+
|
73
|
+
+ return false;
|
74
|
+
|
75
|
+
+}
|
76
|
+
|
77
|
+
|
78
|
+
|
79
|
+
function CreatePdf(){ //当該スプレッドシートをPDFに変換して指定のフォルダに保存するScript
|
80
|
+
|
81
|
+
|
82
|
+
|
83
|
+
var folderId = "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX"; //一時保管用のフォルダIDを定義
|
84
|
+
|
85
|
+
var ss = SpreadsheetApp.getActiveSpreadsheet(); //アクティブなスプレッドシートを取得
|
86
|
+
|
87
|
+
var sheet = ss.getSheetByName(sheetName2); //スプレッドシートの名前を取得
|
88
|
+
|
89
|
+
var LastRow = sheet.getRange(6, 29, 205, 1).getValues().filter(String).length; //空白の要素を除いた取得指定のスプレッドシートの長さを取得
|
90
|
+
|
91
|
+
- var SheetIds = sheet.getRange(6, 29, LastRow, 1).getValues(); //指定範囲の値(シートID)を取得
|
92
|
+
|
93
|
+
+ // LastRowに入っているのは、最終行番号ではなく、シートIDの要素数なので、6を足す必要あり。(途中に空白のセルがない前提ならば本当は5(6-1)でよい。)
|
94
|
+
|
95
|
+
+ var SheetIds = sheet.getRange(6, 29, LastRow + 6, 1).getValues(); //指定範囲の値(シートID)を取得
|
96
|
+
|
97
|
+
var ssId = ss.getId(); //アクティブなスプレッドシートのIDを取得
|
98
|
+
|
99
|
+
var date = new Date(); //今日の日時情報を取得
|
100
|
+
|
101
|
+
var dateString = Utilities.formatDate(date, "Asia/Tokyo", "yyyyMMddHHmmss"); //日時情報を文字列に変換
|
102
|
+
|
103
|
+
|
104
|
+
|
105
|
+
+ var tempFileNames = [] // 一時ファイルのファイル名格納用の配列。
|
106
|
+
|
107
|
+
//表紙の6〜i行目に記載されたSheetIdに基づきPDFを生成する。
|
108
|
+
|
109
|
+
for (var i = 6; i <= SheetIds.length + 6; i++) { //「6」は繰り返しスタートの行数分をプラス
|
110
|
+
|
111
|
+
var shId = sheet.getRange(i, 29).getValue(); //指定シートの指定範囲の値を取得
|
112
|
+
|
113
|
+
+ if (existSheetId(ss, shId) == false){ //シートIDが存在するかどうか判定
|
114
|
+
|
115
|
+
+ Logger.log(`シートID:${shId}(${i}行目)は存在しません。`);
|
116
|
+
|
117
|
+
+ sheet.getRange(i, 30).setValue("シートID不存在"); //シートIDが存在しない場合右隣りのセルに書き込む
|
118
|
+
|
119
|
+
+ continue;
|
120
|
+
|
121
|
+
+ }
|
122
|
+
|
123
|
+
|
124
|
+
|
125
|
+
var name = i + ".pdf"; //保存ファイルの名前を形式指定
|
126
|
+
|
127
|
+
+ tempFileNames.push(name) // 一時ファイル名を格納する。
|
128
|
+
|
129
|
+
var token = ScriptApp.getOAuthToken(); //アクセストークンを取得しAPIリクエストを送信する
|
130
|
+
|
131
|
+
var baseUrl = "https://docs.google.com/spreadsheets/d/" + ssId + "/export?gid=" + shId; //スプレッドシートをPDF出力するためのURLを設定
|
132
|
+
|
133
|
+
var pdfOptions //PDF出力のオプションを設定(URLの組み立て)
|
134
|
+
|
135
|
+
= "&exportFormat=pdf&format=pdf" //ファイル形式の指定 pdf / csv / xls / xlsx
|
136
|
+
|
137
|
+
+ "&size=A4" //用紙サイズ (A4)
|
138
|
+
|
139
|
+
+ "&portrait=true" //用紙の向き true: 縦向き / false: 横向き
|
140
|
+
|
141
|
+
+ "&fitw=true" //ページ幅を用紙にフィットさせるか true: フィットさせる / false: 原寸大
|
142
|
+
|
143
|
+
+ "&top_margin=0.50" //上の余白を設定
|
144
|
+
|
145
|
+
+ "&right_margin=0.50" //右の余白を設定
|
146
|
+
|
147
|
+
+ "&bottom_margin=0.50" //下の余白を設定
|
148
|
+
|
149
|
+
+ "&left_margin=0.50" //左の余白を設定
|
150
|
+
|
151
|
+
+ "&horizontal_alignment=CENTER" //水平方向の位置を設定
|
152
|
+
|
153
|
+
+ "&vertical_alignment=TOP" //垂直方向の位置を設定
|
154
|
+
|
155
|
+
+ "&printtitle=false" //スプレッドシート名の表示有無 trueで表示あり、falseで表示なし(記述方法 "&sheetnames=true")
|
156
|
+
|
157
|
+
+ "&sheetnames=false" //シート名の表示有無 trueで表示あり、falseで表示なし(記述方法 "&sheetnames==true")
|
158
|
+
|
159
|
+
+ "¬e=false" //???
|
160
|
+
|
161
|
+
var url = baseUrl + pdfOptions; //PDFを作成するためのURL
|
162
|
+
|
163
|
+
var options = {headers: {'Authorization': 'Bearer ' + token}}; //headersにアクセストークンを格納する
|
164
|
+
|
165
|
+
var blob = UrlFetchApp.fetch(url, options).getBlob().setName(name); //PDFを作成する
|
166
|
+
|
167
|
+
var folder = DriveApp.getFolderById(folderId); //PDFの保存先フォルダを指定
|
168
|
+
|
169
|
+
folder.createFile(blob) //PDFを指定したフォルダに保存する
|
170
|
+
|
171
|
+
Logger.log(i) //
|
172
|
+
|
173
|
+
var staySecond = 2 //「秒数」を設定(PDFにする処理がHTTP(S)の通信を必要とするため以下の処理が必要)
|
174
|
+
|
175
|
+
Utilities.sleep(staySecond * 1000); //意図しないエラーを防ぐため一時的に処理を遅らせる処理を設定(時間がミリ秒単位のため1000掛ける)
|
176
|
+
|
177
|
+
}
|
178
|
+
|
179
|
+
|
180
|
+
|
181
|
+
//一時保管フォルダのファイル一覧を取得して、1つのPDFに統合する。
|
182
|
+
|
183
|
+
- var iteratorList = folder.getFiles(); //「folder」に対して全てのファイルを取得する
|
184
|
+
|
185
|
+
- var pdfList = []; //リスト用の空の一次元配列を作成
|
186
|
+
|
187
|
+
- var pattern = /.*.pdf$/; //拡張子(ファイルのタイプのこと:「.xlsx」「.pdf」など)の指定(正規表現)
|
188
|
+
|
189
|
+
- while (iteratorList.hasNext()){ //「iteratorList」に対して一時保管フォルダにまだ取り出していないファイルが存在するかを判定
|
190
|
+
|
191
|
+
- var file = iteratorList.next(); //「iteratorList」に対して反復処理でまだ取り出していないファイルがあれば「file」に格納
|
192
|
+
|
193
|
+
- if (pattern.test(file.getName())){ //ファイルの拡張子(ファイルのタイプのこと:「.xlsx」「.pdf」など)をチェック
|
194
|
+
|
195
|
+
- pdfList.push(file); //一次元配列に格納
|
196
|
+
|
197
|
+
- }
|
198
|
+
|
199
|
+
- }
|
200
|
+
|
201
|
+
|
202
|
+
|
203
|
+
+// getFilesだと、作成日時が新しい順に取得してしまい、統合後のページの順番が逆になるため
|
204
|
+
|
205
|
+
+// 一時ファイル名を格納した配列を利用して、出力順に格納する。
|
206
|
+
|
207
|
+
+ var pdfList = [];
|
208
|
+
|
209
|
+
+ for(var i = 0; i < tempFileNames.length; i++){
|
210
|
+
|
211
|
+
+ pdfList.push(folder.getFilesByName(tempFileNames[i]).next());
|
212
|
+
|
213
|
+
+ }
|
214
|
+
|
215
|
+
|
216
|
+
|
217
|
+
var mergedFileName = FileName + dateString + ".pdf"; //結合後のPDFファイルのファイル名を指定
|
218
|
+
|
219
|
+
mergedFile = mergePdfs(folder, mergedFileName, pdfList); //PDFの結合を実行
|
220
|
+
|
221
|
+
- a = merge // 不要なので削除。
|
222
|
+
|
223
|
+
Logger.log("PDFs merged")
|
224
|
+
|
225
|
+
|
226
|
+
|
227
|
+
//一時保管フォルダから統合したPDFを検索して、指定フォルダに移動させる。
|
228
|
+
|
229
|
+
var files = folder.getFilesByName(mergedFileName); //「folder」に対して全ての結合後のPDFファイルのファイル名を「files」に格納
|
230
|
+
|
231
|
+
while (files.hasNext()) { //「filse」に対して一時保管フォルダにまだ取り出していないファイルが存在するかを判定
|
232
|
+
|
233
|
+
var file = files.next();
|
234
|
+
|
235
|
+
var fileName = file.getName(); //「file」に対して格納されている全てのファイル名を「fileName」に格納
|
236
|
+
|
237
|
+
console.log(fileName);
|
238
|
+
|
239
|
+
}
|
240
|
+
|
241
|
+
var storage = DriveApp.getFolderById("XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX"); //結合後のPDFファイルを格納するフォルダIDを設定
|
242
|
+
|
243
|
+
file.moveTo(storage); //「file」に対して作成したファイルを指定先のフォルダに移動させる
|
244
|
+
|
245
|
+
Logger.log("Merged PDF moved")
|
246
|
+
|
247
|
+
|
248
|
+
|
249
|
+
//一時保管フォルダに残された個別PDFを削除する。
|
250
|
+
|
251
|
+
var individualFiles = folder.getFiles(); //「folder」に対して全てのファイル名を取得
|
252
|
+
|
253
|
+
while(individualFiles.hasNext()){ //「individualFiles」に対してまだ取り出していないファイルが存在するかを判定
|
254
|
+
|
255
|
+
var file = individualFiles.next(); //「individualFiles」に対して反復処理でまだ取り出していないファイルがあれば「file」に格納
|
256
|
+
|
257
|
+
file.setTrashed(true); //「file」に対してファイルを削除
|
258
|
+
|
259
|
+
}
|
260
|
+
|
261
|
+
Logger.log("Individual files trashed")
|
262
|
+
|
263
|
+
|
264
|
+
|
265
|
+
//おめでとう
|
266
|
+
|
267
|
+
Browser.msgBox("指定のフォルダにPDFが出力されました") //
|
268
|
+
|
269
|
+
|
270
|
+
|
271
|
+
}
|
272
|
+
|
273
|
+
|
274
|
+
|
275
|
+
function mergePdfs(directory, name, pdf_list) {
|
276
|
+
|
277
|
+
略
|
278
|
+
|
279
|
+
https://qiita.com/mat_aaa/items/d77320769b5ac837a98b に記載のコードを参照のこと。
|
280
|
+
|
281
|
+
}
|
64
282
|
|
65
283
|
```
|
1
エラー処理について追加
test
CHANGED
@@ -8,4 +8,58 @@
|
|
8
8
|
|
9
9
|
|
10
10
|
|
11
|
+
下記のコード例(色つきの追加部分)では、
|
12
|
+
|
13
|
+
指定されたシートIDが存在しない場合、そのシートのPDF出力をスキップし
|
14
|
+
|
15
|
+
シートIDが書かれているセルの右隣に「シート不存在」と書き込み
|
16
|
+
|
17
|
+
次のシートIDの処理に移ります。
|
18
|
+
|
19
|
+
|
20
|
+
|
21
|
+
```diff
|
22
|
+
|
23
|
+
function CreatePdf(){
|
24
|
+
|
25
|
+
|
26
|
+
|
27
|
+
(略)
|
28
|
+
|
29
|
+
|
30
|
+
|
31
|
+
//表紙の6〜i行目に記載されたSheetIdに基づきPDFを生成する。
|
32
|
+
|
33
|
+
for (var i = 6; i <= SheetIds.length + 6; i++) {
|
34
|
+
|
35
|
+
var shId = sheet.getRange(i, 29).getValue(); //指定シートの指定範囲の値を取得
|
36
|
+
|
37
|
+
+ if (!ss.getSheetByName(shId)){ // 指定したシートIDが存在しない場合
|
38
|
+
|
11
|
-
|
39
|
+
+ Logger.log(`シートID:${shId}は存在しません。`);
|
40
|
+
|
41
|
+
+ sheet.getRange(i, 30).setValue("シート不存在"); //シートIDの右隣りのセルに「シート不存在」と書き込む
|
42
|
+
|
43
|
+
+ continue; // PDF書き込み処理をスキップして、次の行に移る
|
44
|
+
|
45
|
+
+ }
|
46
|
+
|
47
|
+
|
48
|
+
|
49
|
+
var name = i + ".pdf";
|
50
|
+
|
51
|
+
var token = ScriptApp.getOAuthToken();
|
52
|
+
|
53
|
+
var baseUrl = "https://docs.google.com/spreadsheets/d/" + ssId + "/export?gid=" + shId;
|
54
|
+
|
55
|
+
var pdfOptions
|
56
|
+
|
57
|
+
|
58
|
+
|
59
|
+
(以下略)
|
60
|
+
|
61
|
+
|
62
|
+
|
63
|
+
|
64
|
+
|
65
|
+
```
|