質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

89.24%

GAS、forms連携させて予約フォームを作成について

受付中

回答 0

投稿 編集

  • 評価
  • クリップ 1
  • VIEW 674

ueshima900

score 24

◆質問内容
フォームの回答によって質問項目を削除したい

◆現状
https://www.lifull.blog/entry/2018/12/10/182307
上を参考にさせていただき作成しております。

①例えば通常の予約数が上限として3件ですが、同じ日時で3回回答がくると、該当の日時が削除され
フォームに反映される
②特定の日時のみ1回の回答で該当の日時を削除

②の部分で、1回目の回答で該当の日時を削除する方法がわからず困っております。

大変わかりにくい質問で申し訳ありませんが、ご教授いただければ幸甚です。

よろしくお願いいたします。

---追記いたします。

var ROOM_MAXIM = 3; // 日程ごとの最大人数
var ARY_CHOICE_VAL = ["1/1 10:00〜","1/1 13:00〜","1/1 15:00〜","1/1 17:00〜","1/1 19:00〜",
"1/2 10:00〜","1/2 13:00〜","1/2 15:00〜","1/2 17:00〜","1/2 17:00"]
var CHOICE_LENGTH = ARY_CHOICE_VAL.length; // 選択肢の数
var TMP_NEW_CHOICE_VAL = "全日程満員です。次の設問で希望の日程を入力してください"; // 通常選択肢が全て無くなった後に追加する選択肢VALUE
var STR_TARGET_TITLE = "希望日"; // 処理の対象とする質問タイトル名

var TARGET_CHOICE_VAL = ["1/1 10:00〜","1/1 17:00〜","1/2 10:00〜"] //この配列内のアイテムは1件で質問を消す

/**

  • フォーム送信時イベント(初期処理)
    */
    function onSend(e) {
    reBuildItem(e.response.getItemResponses()); //functionの引数eからイベント発生時の回答時のレスポンス取得して受け渡し
    }

/**

  • 選択肢の再構築
    • @param {array} 選択肢ごとの回答カウントデータ配列
      */ 
      function reBuildItem(rAnswer) {
      var form = FormApp.getActiveForm(); //アクティブなformオブジェクトを取得
      var items = form.getItems(); // フォームのUI項目を取得
      var allAnswer = getAnswer(); // 全回答の取得
      Logger.log(allAnswer)
      // 今回の回答を取得(※プルダウンUIで未選択の場合は回答データに含まれない模様です※)
      var nowAnswerVal = rAnswer[0].getResponse();
      Logger.log(nowAnswerVal)
      var nowAnswerTitle = rAnswer[0].getItem().getTitle();

if (nowAnswerTitle == STR_TARGET_TITLE) { // 「希望日程」プルダウンで未選択の場合は何もしない(項目タイトルで判断)

for (var i = 0; i < items.length; i++) { // UI項目分ループ
var item = items[i];
var itemTitle = String(item.getTitle());
var itemType = String(item.getType()); // UI項目のタイプ

if (itemType == "LIST" && itemTitle == STR_TARGET_TITLE) { // リストボックスかつ項目タイトルが対象の場合
var choiceArray = [];
var arrayCount = 0
var flgChoiceNone = false; // 回答突き合わせ後選択肢が残るか否かのフラグ

for (var j = 0; j < CHOICE_LENGTH; j++) { // リスト選択肢分ループ
var choiceVal = ARY_CHOICE_VAL[j];
var flgSet = false; // 上限内か否かのフラグ

if (flgChoiceNone == false) { // 選択肢がまだ残っている場合
if (j == 0 && nowAnswerVal == TMP_NEW_CHOICE_VAL) { // 選択肢が全て無くなった後の追加選択肢のチェックはスルーする
flgChoiceNone = true;
} else {
if (allAnswer[j] < ROOM_MAXIM || allAnswer[3] == ROOM_MAXIM_EXC) { // 回答上限チェック ここで特定の値
flgSet = true;

}
}
}
if (flgSet == true) { // 回答上限に達してない選択肢は退避
choiceArray[arrayCount] = choiceVal;
arrayCount++;
} else {
if (choiceVal == nowAnswerVal) { // 今回の回答だった場合
Logger.log('上限に達した');
}
}
}

if (arrayCount == 0) { // 選択肢が全てなくなった場合
choiceArray[arrayCount] = TMP_NEW_CHOICE_VAL; // 追加選択肢をセット
}
// リストボックス作り直し
item.asListItem().setChoiceValues(choiceArray);
}
}
}
}

/**

  • 回答データの取得
  • 回答データが取得できない(例外の場合)エラースロー
    • @return {array} 選択肢ごとの回答カウントデータ配列 
      */
      function getAnswer() {
      var form = FormApp.getActiveForm();
      var formResponses = form.getResponses(); // 全回答内容を取得
      var aryAnswerCount = []; // 選択肢分配列定義
      for (var k = 0; k < CHOICE_LENGTH; k++) { // 配列をゼロで初期化(fillメソッド使えないので)
      aryAnswerCount[k] = 0;
      }
      var flgAnswer = false;

for (var i = 0; i < formResponses.length; i++) {
flgAnswer = true;
var formResponse = formResponses[i]; // 回答ひとつ分を取得
var itemResponses = formResponse.getItemResponses(); // 質問項目を取得

for (var j = 0; j < itemResponses.length; j++) { // 回答内容をひとつずつチェック
var itemResponse = itemResponses[j];
var question = itemResponse.getItem().getTitle();
var answer = itemResponse.getResponse();

if (question == STR_TARGET_TITLE) { // 項目タイトルが対象の場合
var x = ARY_CHOICE_VAL.indexOf(answer);
if (x != -1) { // 選択肢ごとに回答数カウント
aryAnswerCount[x]++;
}
}
}
}

if (flgAnswer == false) { // 例外:回答がまだなければエラー吐いて終了
throw new Error("not Answer"); 
} else {
return aryAnswerCount;
}
}

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正の依頼

  • papinianus

    2019/08/13 18:59

    1ができている現状のコードを書いてください。

    キャンセル

  • ueshima900

    2019/08/14 09:44

    ご返信ありがとうございます。

    参考サイトのものを配列だけ配列だけ変更しているだけで、他は全くの流用になっておりますが、記載させていただきます。
    変更箇所
    ・var ARY_CHOICE_VAL
    追加
    ・var TARGET_CHOICE_VAL この配列の内のアイテムは1件で質問を消すようにしたいです。

    // FormApp.getActiveForm()

    var ROOM_MAXIM = 3; // 日程ごとの最大人数
    var ARY_CHOICE_VAL = ["1/1 10:00〜","1/1 13:00〜","1/1 15:00〜","1/1 17:00〜","1/1 19:00〜",
    "1/2 10:00〜","1/2 13:00〜","1/2 15:00〜","1/2 17:00〜","1/2 17:00"]
    var CHOICE_LENGTH = ARY_CHOICE_VAL.length; // 選択肢の数
    var TMP_NEW_CHOICE_VAL = "全日程満員です。次の設問で希望の日程を入力してください"; // 通常選択肢が全て無くなった後に追加する選択肢VALUE
    var STR_TARGET_TITLE = "希望日"; // 処理の対象とする質問タイトル名

    var TARGET_CHOICE_VAL = ["1/1 10:00〜","1/1 17:00〜","1/2 10:00〜"] //この配列内のアイテムは1件で質問を消す

    /**
    * フォーム送信時イベント(初期処理)
    */
    function onSend(e) {
    reBuildItem(e.response.getItemResponses()); //functionの引数eからイベント発生時の回答時のレスポンス取得して受け渡し
    }

    /**
    * 選択肢の再構築
    *
    * @param {array} 選択肢ごとの回答カウントデータ配列
    */
    function reBuildItem(rAnswer) {
    var form = FormApp.getActiveForm(); //アクティブなformオブジェクトを取得
    var items = form.getItems(); // フォームのUI項目を取得
    var allAnswer = getAnswer(); // 全回答の取得
    Logger.log(allAnswer)
    // 今回の回答を取得(※プルダウンUIで未選択の場合は回答データに含まれない模様です※)
    var nowAnswerVal = rAnswer[0].getResponse();
    Logger.log(nowAnswerVal)
    var nowAnswerTitle = rAnswer[0].getItem().getTitle();

    if (nowAnswerTitle == STR_TARGET_TITLE) { // 「希望日程」プルダウンで未選択の場合は何もしない(項目タイトルで判断)

    for (var i = 0; i < items.length; i++) { // UI項目分ループ
    var item = items[i];
    var itemTitle = String(item.getTitle());
    var itemType = String(item.getType()); // UI項目のタイプ

    if (itemType == "LIST" && itemTitle == STR_TARGET_TITLE) { // リストボックスかつ項目タイトルが対象の場合
    var choiceArray = [];
    var arrayCount = 0
    var flgChoiceNone = false; // 回答突き合わせ後選択肢が残るか否かのフラグ

    for (var j = 0; j < CHOICE_LENGTH; j++) { // リスト選択肢分ループ
    var choiceVal = ARY_CHOICE_VAL[j];
    var flgSet = false; // 上限内か否かのフラグ

    if (flgChoiceNone == false) { // 選択肢がまだ残っている場合
    if (j == 0 && nowAnswerVal == TMP_NEW_CHOICE_VAL) { // 選択肢が全て無くなった後の追加選択肢のチェックはスルーする
    flgChoiceNone = true;
    } else {
    if (allAnswer[j] < ROOM_MAXIM || allAnswer[3] == ROOM_MAXIM_EXC) { // 回答上限チェック ここで特定の値
    flgSet = true;

    }
    }
    }
    if (flgSet == true) { // 回答上限に達してない選択肢は退避
    choiceArray[arrayCount] = choiceVal;
    arrayCount++;
    } else {
    if (choiceVal == nowAnswerVal) { // 今回の回答だった場合
    Logger.log('上限に達した');
    }
    }
    }

    if (arrayCount == 0) { // 選択肢が全てなくなった場合
    choiceArray[arrayCount] = TMP_NEW_CHOICE_VAL; // 追加選択肢をセット
    }
    // リストボックス作り直し
    item.asListItem().setChoiceValues(choiceArray);
    }
    }
    }
    }




    /**
    * 回答データの取得
    * 回答データが取得できない(例外の場合)エラースロー
    *
    * @return {array} 選択肢ごとの回答カウントデータ配列
    */
    function getAnswer() {
    var form = FormApp.getActiveForm();
    var formResponses = form.getResponses(); // 全回答内容を取得
    var aryAnswerCount = []; // 選択肢分配列定義
    for (var k = 0; k < CHOICE_LENGTH; k++) { // 配列をゼロで初期化(fillメソッド使えないので)
    aryAnswerCount[k] = 0;
    }
    var flgAnswer = false;

    for (var i = 0; i < formResponses.length; i++) {
    flgAnswer = true;
    var formResponse = formResponses[i]; // 回答ひとつ分を取得
    var itemResponses = formResponse.getItemResponses(); // 質問項目を取得

    for (var j = 0; j < itemResponses.length; j++) { // 回答内容をひとつずつチェック
    var itemResponse = itemResponses[j];
    var question = itemResponse.getItem().getTitle();
    var answer = itemResponse.getResponse();

    if (question == STR_TARGET_TITLE) { // 項目タイトルが対象の場合
    var x = ARY_CHOICE_VAL.indexOf(answer);
    if (x != -1) { // 選択肢ごとに回答数カウント
    aryAnswerCount[x]++;
    }
    }
    }
    }

    if (flgAnswer == false) { // 例外:回答がまだなければエラー吐いて終了
    throw new Error("not Answer");
    } else {
    return aryAnswerCount;
    }
    }


    nowAnswerValがTARGET_CHOICE_VALに含まれている場合
    choiceArrayのリストから削除とかしたかったのですが、うまくいかずです。。

    キャンセル

  • papinianus

    2019/08/14 18:24

    大変お手数ですが、質問に追記していただけませんか?

    キャンセル

  • ueshima900

    2019/08/14 18:39

    申し訳ありません。追記させていただきました。

    キャンセル

まだ回答がついていません

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 89.24%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる