LoginSignup
1
2

More than 5 years have passed since last update.

Lightning componentでteamspiritの勤怠打刻を行う

Last updated at Posted at 2018-11-07

チームスピリット使ってますか?弊社では勤怠打刻と経費精算で使ってます。便利ですね。
さて、勤務時間の把握のために勤怠打刻を使っているという組織は少なくないと思うのですが、如何せんteamspiritデフォルトのコンポーネントは表示が大きすぎると思ってはいませんか?僕は思っています。
そこで、オレオレ打刻lightning componentを作ろうということで、この項目かなーと適当に作ってみたら、どうやら違うみたいでエラーになります。困ったなと思っていた時に出会ったのがこちら

Slack のコマンドを使って TeamSpirit に打刻する
ついでにこれも実装しようと思ったらうまくいきませんでした(悲

ソースはgithubに置いてありますので、そちらを参考にさせていただきました。

コンポーネントはシンプルな1ボタンのみ

ts打刻.jpg
出社打刻が無いと怒られる仕様です

yourAttendance.cmp

yourAttendance.cmp
<aura:component controller="yourAttendanceController" implements="flexipage:availableForAllPageTypes" access="global" >
    <aura:handler name="init" value="{!this}" action="{!c.doInit}"/>
    <aura:attribute name="showOrHide" type="object[]"/> 
    <aura:attribute name="attendanceString" type="string[]"/> 
    <aura:if isTrue="{!v.showOrHide[0]}">
        <article class="slds-card">
          <div class="slds-card__header slds-grid">
            <header class="slds-media slds-media_center slds-has-flexi-truncate">
              <div class="slds-media__figure">
                <span class="slds-icon_container slds-icon-standard-account" title="account">
                  <span class="slds-assistive-text">勤怠打刻</span>
                </span>
              </div>
              <div class="slds-media__body">
                <h2 class="slds-card__header-title">
                    <span>勤怠打刻</span>
                </h2>
              </div>
            </header>
          </div>
          <div class="slds-card__body slds-card__body_inner">
                <aura:if isTrue="{! empty(v.attendanceString[0])}"><!--ボタン押したらボタンが消える処理-->
                  <button class="slds-button slds-button_brand" onclick="{!c.updateAttendance}">{!v.showOrHide[1]}</button>
                  <aura:unescapedHtml value="{!v.showOrHide[2]}" />
              </aura:if>
              <aura:unescapedHtml value="{!v.attendanceString[1]}" />
              <!-- debug用
              showOrHide0: {!v.showOrHide[0]}<br />
              showOrHide1: {!v.showOrHide[1]}<br />
              showOrHide2: {!v.showOrHide[2]}<br />
              showOrHide3 empId: {!v.showOrHide[3]}<br />
              showOrHide4 yearMonth: {!v.showOrHide[4]}<br />
              showOrHide5 startDate : {!v.showOrHide[5]}<br />
              showOrHide6 lastModifiedDate: {!v.showOrHide[6]}<br />
              showOrHide7 stdStartTime: {!v.showOrHide[7]}<br />
              showOrHide8 stdEndTime: {!v.showOrHide[8]}<br />
              showOrHide9 empToday: {!v.showOrHide[9]}<br />
              {!v.attendanceString[0]}<br />
              {!v.attendanceString[1]}<br />
              {!v.attendanceString[2]}<br />
              {!v.attendanceString[3]}<br />
              {!v.attendanceString[4]}<br />
              {!v.attendanceString[5]}<br />
              {!v.attendanceString[6]}<br />
              {!v.attendanceString[7]}<br />
              {!v.attendanceString[8]}<br />
              {!v.attendanceString[9]}<br />
              {!v.attendanceString[10]}<br />
                -->
          </div>
        </article>
    </aura:if>


</aura:component>

yourAttendanceController.js

yourAttendanceController.js

({
    doInit : function(component, event, helper) {
        helper.isSetYourTime(component);
    },
    updateAttendance : function(component, event, helper)  {
        helper.updateAttendance_helper(component) ;
    },
})

yourAttendanceHelper.js

yourAttendanceHelper.js
({
    isSetYourTime : function(component) {
        var action = component.get("c.isSetMyAttendance");
        action.setCallback(this, function(response) {
            var state = response.getState();
            console.log("state: " + state);
            if (state === "SUCCESS") {
                component.set("v.showOrHide", response.getReturnValue());
            //ここまで state success
            }
            else {
                console.log("Failed with state?: " + state);
            }
        });        
        // Send action off to be executed
        $A.enqueueAction(action); 
        console.log("action: " + action);    
    },
    updateAttendance_helper : function(component) {//出勤打刻
        var save_action = component.get("c.getInsertAttendance");
        //console.log('showhideのhelper ' + showOrHide);
        /*
         * var dataJSON = [{
            empId : showOrHide[3],
            yearMonth : showOrHide[4],
            startDate : showOrHide[5],
            lastModifiedDate : showOrHide[6],
            stdStartTime : showOrHide[7],
            stdEndTime : showOrHide[8],
            empToday : showOrHide[9]
        }];
        //JSONにしてStringで渡さないと受け取ってもらえない。Apex側でdeserializeする        
        save_action.setParams({
            "paramsJSON": JSON.stringify(dataJSON),
        });        
        console.log('paramsJSON ' + JSON.stringify(dataJSON));
        */

        save_action.setCallback(this, function(response) {
            var state = response.getState();
            console.log("state: " + state);
            if (state === "SUCCESS") {
                component.set("v.attendanceString", response.getReturnValue());
            //ここまで state success
            }
            else {
                console.log("出勤ボタンエラー: " + state);
            }
        });        
        // Send action off to be executed
        $A.enqueueAction(save_action);
    },
})

yourAttendanceController.apxc

yourAttendanceController.apxc
public class yourAttendanceController {
    //初期データから表示非表示を判定する
    @AuraEnabled
    public static LIST<object> isSetMyAttendance() {
        //初期データの取得
        loadDataResponseData ld = loadData() ;
        //初期データの配置
        List<teamspirit__AtkEmpDay__c> empDays = ld.empDays;//勤怠日次リスト
        String empId = ld.empId;
        Integer yearMonth = ld.yearMonth;
        String startDate = ld.startDate;
        String lastModifiedDate = ld.lastModifiedDate;
        Integer stdStartTime = ld.stdStartTime;
        Integer stdEndTime = ld.stdEndTime;
        teamspirit__AtkEmpDay__c empToday = ld.empToday; //今日の勤怠日次   
        Map<String, Object> lastData = ld.lastData;
        //初期データのここまで

        //本日の状況を取得
        TimeTableResponse res = new TimeTableResponse();
        timeTableResponseData ttrd = new timeTableResponseData();
        //本日の状況を取得
        List<Map<String, Integer>> timeTable = new List<Map<String, Integer>>();
        ttrd = getTimeTable(res,empToday,timeTable);        
        res.timeTable = timeTable;
        system.debug('タイムテーブルはとれているか ' + timeTable);
        //取れてた:13:27:32:491 USER_DEBUG [50]|DEBUG|タイムテーブルはとれているか ({from=null, to=null, type=1})

        system.debug('レス' + res );
        //18:31:38:491 USER_DEBUG [46]|DEBUG|res TimeTableResponse:[isHoliday=false, timeTable=({from=540, to=1080, type=1}, {from=690, to=780, type=21})] 出退勤両方ある
        //18:33:06:540 USER_DEBUG [46]|DEBUG|res TimeTableResponse:[isHoliday=false, timeTable=({from=null, to=null, type=1})] 出退勤両方無い
        //18:33:50:547 USER_DEBUG [46]|DEBUG|res TimeTableResponse:[isHoliday=false, timeTable=({from=540, to=null, type=1}, {from=690, to=780, type=21})] 出勤だけある
        //18:34:29:491 USER_DEBUG [46]|DEBUG|res TimeTableResponse:[isHoliday=false, timeTable=({from=540, to=1065, type=1}, {from=690, to=780, type=21})]
        //18:36:25:629 USER_DEBUG [46]|DEBUG|res TimeTableResponse:[isHoliday=false, timeTable=({from=480, to=null, type=1}, {from=690, to=780, type=21})] 8時出勤のみ
        //18:37:24:551 USER_DEBUG [46]|DEBUG|res TimeTableResponse:[isHoliday=false, timeTable=({from=480, to=1200, type=1}, {from=690, to=780, type=21})] 8時出勤20時退社
        //18:38:40:491 USER_DEBUG [46]|DEBUG|res TimeTableResponse:[isHoliday=false, timeTable=({from=660, to=null, type=1}, {from=690, to=780, type=21})] 11時出社のみ
        // timeTable[1]は裁量労働の基本労働時間だと思う。設定してあると変化しない
        system.debug('レスres.get(timeTable) ' + res.timeTable);        
        system.debug('res.get(timeTable)from ' + res.timeTable[0].get('from'));
        //出退勤両方無い場合は res.timeTable == NULL
        //出勤だけある場合は  res.timeTable[0].get('from') != NULL && res.timeTable[0].get('to') == NULL
        //両方ある場合は res.timeTable[0].get('from') != NULL && res.timeTable[0].get('to') != NULL

        //現在時刻の取得
        Datetime now = Datetime.now();
        Integer nowHour = now.hour();



        //カードの出力
        //両方ある場合は非表示にする

        LIST<object> showOrHide = new LIST<object>();
        //showOrHide[0]の設定ここから:card全体の表示非表示の切り替え
        if(nowHour<=15){// 〜15:59
            if(res.timeTable[0].get('from') != NULL){// 〜15:59で出勤打刻がある場合
                showOrHide.add('false');//cardを消す
            }else{
                showOrHide.add('true');//cardを表示
            }
        }else{// 15:59〜
            if(res.timeTable[0].get('to') != NULL){// 15:59〜で退勤打刻がある場合
                showOrHide.add('false');//cardを消す
            }else{
                showOrHide.add('true');//cardを表示
            }
        }
        //showOrHide[0]の設定ここまで

        //showOrHide[1]と[2]の設定ここから
        //ボタンの表示文言
        if(nowHour<=15){//〜15:59
            showOrHide.add('出勤打刻する');
            showOrHide.add('');//showOrHide[2]
        }else{
            if(res.timeTable[0].get('from') != NULL){// 〜15:59で出勤打刻がある場合
                showOrHide.add('退勤打刻する');       
                showOrHide.add('');//showOrHide[2]
            }else{
                showOrHide.add('退勤打刻する');
                showOrHide.add('<p class="slds-text-color_error">出社打刻がありませんでした。明日はよろしくおねがいします。</p>');//showOrHide[2]
            }
        } 
        //showOrHide[1]の設定ここまで

        //debug用
        showOrHide.add(empId);
        showOrHide.add(yearMonth);
        showOrHide.add(startDate);
        showOrHide.add(lastModifiedDate);
        showOrHide.add(stdStartTime);
        showOrHide.add(stdEndTime);
        showOrHide.add(empToday);
        //debug用

        return showOrHide;        
    }

    public class TimeTableResponse {
        public List<Map<String, Integer>> timeTable;
        public Boolean isHoliday;
    }

    //出勤打刻
    @AuraEnabled
    public static LIST<String> getInsertAttendance() {
        //初期データの取得
        loadDataResponseData ld = loadData() ;
        //初期データの配置
        List<teamspirit__AtkEmpDay__c> empDays = ld.empDays;//勤怠日次リスト
        String empId = ld.empId;
        Integer yearMonth = ld.yearMonth;
        String startDate = ld.startDate;
        String lastModifiedDate = ld.lastModifiedDate;
        Integer stdStartTime = ld.stdStartTime;
        Integer stdEndTime = ld.stdEndTime;
        teamspirit__AtkEmpDay__c empToday = ld.empToday; //今日の勤怠日次   
        Map<String, Object> lastData = ld.lastData;
        //初期データのここまで

        //getTimeTable タイムテーブルを取得する
        timeTableResponseData ttrd = new timeTableResponseData();
        TimeTableResponse resTmp = new TimeTableResponse();
        List<Map<String, Integer>> timeTable = new List<Map<String, Integer>>();
        ttrd = getTimeTable(resTmp,empToday,timeTable); 

        //結果出力用
        LIST<String> insertAttendance = new LIST<String>();

        //inputTimeTable
        Map<String, Object> params = getBaseParams( empId, yearMonth, startDate,lastModifiedDate);
        //inputTimeTable


        //setAttendance
        DateTime now = DateTime.now();
        Integer timeHM = now.hour() * 60 + now.minute();
        //出勤 or 退勤
        //現在時刻の取得
        Integer nowHour = now.hour();
        Integer attendance;
        if(nowHour<=15){//〜15:59
            attendance = 0;
        }else{
            attendance = 1;
        }

        Map<String, Object> input = new Map<String, Object>{'comment' => '', 'time' => timeHM, 'face' => attendance, 'fix' => false, 'type' => 10};
        params.put('input', input);
        params.put('prevFlag', false);
        params.put('stdStartTime', stdStartTime);
        params.put('stdEndTime', stdEndTime);
        String jsonReqSetAttendance = JSON.serialize(params);
        system.debug('jsonReqSetAttendance ' + jsonReqSetAttendance);
        Map<String, Object> resSetAttendance = teamspirit.RtkPotalCtl.inputTime(jsonReqSetAttendance);
        lastModifiedDate = String.valueOf(resSetAttendance.get('lastModifiedDate'));

        if(resSetAttendance.get('result') == 'OK'){
            insertAttendance.add('false');//cardの非表示
            if(nowHour<=15){//〜15:59
                insertAttendance.add('<p class="slds-text-color_success">👍出勤登録が完了しました👍</p>');
            }else{
                insertAttendance.add('<p class="slds-text-color_success">😀😌退勤登録が完了しました👍<br />今日も一日お疲れ様でした!🍻</p>');                
            }
        }else{
            insertAttendance.add('false');//cardの非表示
            if(nowHour<=15){//〜15:59
                insertAttendance.add('<p class="slds-text-color_error">😱😱出勤登録が失敗しました。Georgeまでお知らせください' + resSetAttendance + '</p>');
            }else{
                insertAttendance.add('<p class="slds-text-color_error">😱😱退勤登録が失敗しました。Georgeまでお知らせください' + resSetAttendance + '</p>');                
            }
        }
        //setAttendanceここまで
        return insertAttendance;
    }

    //https://github.com/ngs/ts-dakoku/blob/master/apex/src/classes/TSTimeTableAPIController.cls

    //loadDataで返ってくるデータを入れる
    public class loadDataResponseData {
        public Map<String, Object> lastData;
        public List<teamspirit__AtkEmpDay__c> empDays;
        public String empId;
        public Integer yearMonth;
        public String startDate;
        public String lastModifiedDate;
        public Integer stdStartTime;
        public Integer stdEndTime;
        public teamspirit__AtkEmpDay__c empToday;
    }

    //初期データ取得
    @AuraEnabled
    public static loadDataResponseData loadData() {
        loadDataResponseData res = new loadDataResponseData();
        res.lastData = teamspirit.RtkPotalCtl.getLastModifiedDate();//自分の最終更新データが取得できる模様
        res.empId = (String) res.lastData.get('empId');//社員IDを取得
        res.lastModifiedDate = String.valueOf(res.lastData.get('lastModifiedDate'));//最終更新日を取得
        Map<String, Object> empMonth = teamspirit.RtkPotalCtl.loadEmpMonth('');//当月のデータの取得?
        List<teamspirit__AtkConfig__c > configs = (List<teamspirit__AtkConfig__c>) empMonth.get('configs');//empMonthの中にいろんなデータが入っている模様。configを取ってくる
        teamspirit__AtkConfig__c config = configs != null && configs.size() > 0 ? configs[0] : null;
        if (config != null) {
            res.stdStartTime = Integer.valueOf(config.teamspirit__StdStartTime__c);
            res.stdEndTime = Integer.valueOf(config.teamspirit__StdEndTime__c);
        }
        res.empDays = (List<teamspirit__AtkEmpDay__c>) empMonth.get('empDays');//勤怠日次のリストがempDaysでマップされて入っている模様
        if (res.empDays == null) {//empDaysが未作成だった場合は作成するっぽい
            res.empDays = new List<teamspirit__AtkEmpDay__c>();
        }
        Date today = getToday();
        for (teamspirit__AtkEmpDay__c day: res.empDays) {
            if (day.teamspirit__Date__c == today) {//今日判定をして今日の日付が出たらempTodayをdayに入れてbreak
                res.empToday = day;
                break;
            }
        }
        res.yearMonth = (Integer) empMonth.get('yearMonth');
        res.startDate = (String) empMonth.get('startDate');

        return res;
    }


    //loadDataで返ってくるデータを入れる
    public class timeTableResponseData {
        public TimeTableResponse ttr;
        public List<Map<String, Integer>> timeTable;
    }

    //TimeTableのデータを取得する
    public static timeTableResponseData getTimeTable(TimeTableResponse res,teamspirit__AtkEmpDay__c empToday,List<Map<String, Integer>> timeTable) { //勤怠日次>時間配分の取得
        timeTableResponseData ttrd = new timeTableResponseData();
        res.isHoliday = isHoliday(empToday);
        res.timeTable = timeTable;
        if (empToday == null) {
            ttrd.ttr = res;
            return ttrd;
        }

        Map<String, Integer> item = new Map<String, Integer>();
        item.put('from', Integer.valueOf(empToday.teamspirit__StartTime__c));
        item.put('to', Integer.valueOf(empToday.teamspirit__EndTime__c));
        item.put('type', 1);
        timeTable.add(item);

        List<String> timeTableStrItems = (empToday.teamspirit__TimeTable__c == null ? '' : empToday.teamspirit__TimeTable__c).split(':');
        for (String timeTableStr: timeTableStrItems) {
            if(timeTableStr == '') {
                continue;
            }
            item = new Map<String, Integer>{};

                String str = timeTableStr.substring(0, 4);
            str = timeTableStr.substring(0, 4);
            if (str != '----') {
                item.put('from', Integer.valueOf(str));
            }
            str = timeTableStr.substring(4, 8);
            if (str != '----') {
                item.put('to', Integer.valueOf(str));
            }
            str = timeTableStr.substring(8, 10);
            item.put('type', Integer.valueOf(str));

            timeTable.add(item);
        }

        ttrd.ttr = res;
        ttrd.timeTable= timeTable;
        return ttrd;
    }    



    private static Date getToday() {
        return Date.today();
    }

    public static Boolean isHoliday(teamspirit__AtkEmpDay__c empToday) {
        return empToday != null && empToday.teamspirit__DayType__c != null && Integer.valueOf(empToday.teamspirit__DayType__c) > 0 && empToday.teamspirit__HolidayWorkApplyId__c == null;
    }

    private static Map<String, Object>getBaseParams(String empId,Integer yearMonth,String startDate,String lastModifiedDate) {
        Map<String, Object> params = new Map<String, Object>();
        params.put('empId', empId);
        params.put('month', yearMonth);
        params.put('startDate', startDate);
        params.put('lastModifiedDate', lastModifiedDate);
        params.put('date', DateTime.newInstance(getToday(), Time.newInstance(0, 0, 0, 0)).format('yyyy-MM-dd'));
        return params;
    }
}

yourAttendanceControllerTEST.apxc

yourAttendanceControllerTEST.apxc
@isTest
public class yourAttendanceControllerTest {
    @isTest(Seealldata=true)
    static void testMethod1() {
        yourAttendanceController.isSetMyAttendance();
        yourAttendanceController.getInsertAttendance();
    }
}

打刻が終わるとこんな感じの表示に変わります

dakokuDone.jpg
0時〜15:59までは出勤打刻ボタンを表示。
16:00以降は退勤ボタンを表示。
各時間で打刻が終わっていると、勤怠打刻の表示自体が非表示になります。

コードはgithubへ

geeorgey/ts-dakoku-lightningcomponent

更新履歴

2018.11.08 yourAttendanceController.apxc を更新

1
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
2