Help us understand the problem. What is going on with this article?

ThingWorx ウィジェット開発 その4 機能を拡張する

More than 1 year has passed since last update.

はじめに

この記事は、「ThingWorx ウィジェット開発 その3 マッシュアップへの組み込み」からの続きです。

目次

さて、これまでにアナログ時計の機能を持った Analog Clock ウィジェットを作成する方法を見てきました。今回は作ったアナログ時計をアラーム時計に進化させます。アラーム時計としての仕様は下記の通りです。

  • アラーム時刻をセットする
  • アラーム状態を継続する期間をセットする
  • アラーム時刻になると、イベントを発行する
  • アラーム時刻になると、文字盤を赤に変える
  • アラーム状態を解除する時間になったら、通常の文字盤に戻す

さっそくコードを見ていきましょう。

第十二歩:マッシュアップビルダー用の改造

今回も先にコードの全体を見渡してみましょう。

analog_clock.ide.js
TW.IDE.Widgets.analog_clock = function () {

    this.widgetIconUrl = function() {
        return  "'../Common/extensions/AnalogClock/ui/analog_clock/default_widget_icon.ide.png'";
    };

    this.widgetProperties = function () {
        return {
            'name': 'Analog Clock',
            'description': '',
            'category': ['Common'],
            'isResizable': true,
            'supportsAutoResize': true,
            'properties': {
                'date' : {
                    'baseType': 'DATETIME',
                    'isBindingTarget' : true,
                    'isVisible': true
                },
                'alarm date': {
                    'baseType': 'DATETIME',
                    'isBindingTarget': true,
                    'isVisible': true
                },
                'alarm': {
                    'baseType': 'BOOLEAN',
                    'defaultValue': false,
                    'isEditable': true,
                    'isBindingTarget': true,
                    'isVisible': true
                },
                'reset after': {
                    'baseType': 'INTEGER',
                    'defaultValue': 300,
                    'isEditable': true,
                    'isBindingTarget': true,
                    'isVisible': true
                },
                'Width': {
                    'description': TW.IDE.I18NController.translate('tw.textbox-ide.properties.width.description'),
                    'defaultValue': 120,
                    'isEditable': true,
                    'isVisible': true
                },
                'Height': {
                    'description': TW.IDE.I18NController.translate('tw.textbox-ide.properties.height.description'),
                    'defaultValue': 120,
                    'isEditable': true,
                    'isVisible': true
                }
            }
        }
    };

    this.afterSetProperty = function (name, value) {
        var thisWidget = this;
        var refreshHtml = false;
        switch (name) {
            case 'Style':
            case 'Width':
                this.setProperty('Height', value);
            case 'Height':
                this.setProperty('Width', value);
            case 'Alignment':
                refreshHtml = true;
                break;
            default:
                break;
        }
        return refreshHtml;
    };

    this.renderHtml = function () {
        // return any HTML you want rendered for your widget
        // If you want it to change depending on properties that the user
        // has set, you can use this.getProperty(propertyName).
        return  '<div class="widget-content widget-analog_clock">' +
                    '<img class="aClock" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Back.svg"> ' +
                    '<img class="aClock aClock_shortArm" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_ShortArm.svg" />' +
                    '<img class="aClock aClock_longArm" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_LongArm.svg" />' +
                    '<img class="aClock" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Center.svg" />' +
                '</div>';
    };

    this.afterRender = function () {
        // NOTE: this.jqElement is the jquery reference to your html dom element
        //       that was returned in renderHtml()
    };

    this.widgetEvents = function() {
        return {
            'Alarmed': {}
        };
    };


};

変更点として、alarm datealarmreset afterという三つのプロパティを新たに定義しました。これらのプロパティの使い道は下記の通りです。

alarm date
DATETIME 型。ここに指定した日付・時間になると、Analog Clock ウィジェットをアラーム状態にする。

alarm
BOOLEAN 型。true を返すときには、アラーム状態にする。false を返す時には、たとえalarm dateがセットされていてもアラーム状態にしない。

reset after
INTEGER 型。ここに指定した秒数を経過すると、アラーム状態を解除する。

また、下記のコードで、このウィジェットがalarmedというイベントを発行することを指定しています。

analog_clock.ide.js
    this.widgetEvents = function() {
        return {
            'Alarmed': {}
        };
    };

analog_clock.ide.js の変更点は以上ですね。最終的にウィジェットをインポートした後のプロパティペインは下図のようになります。
image.png

第十三歩:ランタイム用の改造

次に、ランタイム用のコードを見ていきましょう。まずは全体を。

analog_clock.runtime.js
TW.Runtime.Widgets.analog_clock= function () {
    var shortArm;
    var longArm;
    var hour;
    var minute;
    var second;
    var basePlate;
    var shortArmRotate;
    var longArmRotate;
    var currentDate;
    var alarmDate;
    var isAlarmSet;
    var hasGoneOff;
    var resetAfter;

    this.renderHtml = function () {
        return  '<div class="widget-content widget-analog_clock">' +
                    '<img class="aClock aClock_base"     src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Back.svg" />' +
                    '<img class="aClock aClock_shortArm" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_ShortArm.svg" />' +
                    '<img class="aClock aClock_longArm"  src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_LongArm.svg" />' +
                    '<img class="aClock" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Center.svg" />' +
                '</div>';
    };

    this.afterRender = function () {
        basePlate = this.jqElement.find('.aClock_base');
        shortArm = this.jqElement.find('.aClock_shortArm');
        longArm = this.jqElement.find('.aClock_longArm');
        alarmDate = this.getProperty('alarm date');
        isAlarmSet = this.getProperty('alarm');
        hasGoneOff = false;
        resetAfter = this.getProperty('reset after');
    };

    this.updateProperty = function (updatePropertyInfo) {

        if (updatePropertyInfo.TargetProperty === 'alarm') {
            if (updatePropertyInfo.SinglePropertyValue === 'false') {
                isAlarmSet = false;
            } else {
                isAlarmSet = true;
            }
            this.setProperty('alarm', isAlarmSet);
        }

        if (updatePropertyInfo.TargetProperty === 'alarm date') {
            alarmDate = new Date(updatePropertyInfo.SinglePropertyValue);
            this.setProperty('alarm date', alarmDate);
        }

        if (updatePropertyInfo.TargetProperty === 'reset after') {
            resetAfter = parseInt(updatePropertyInfo.SinglePropertyValue, 10);
            this.setProperty('reset after', resetAfter);
        }

        if (updatePropertyInfo.TargetProperty === 'date') {

            currentDate = new Date(updatePropertyInfo.SinglePropertyValue);
            hour = currentDate.getHours();
            minute = currentDate.getMinutes();
            second = currentDate.getSeconds();

            if (hour > 11) {
                hour = hour - 12;
            } else if (!hour) {
                hour = 0;
            }

            if (minute === '60') {
                minute = 0;
            } else if (!minute) {
                minute = 0;
            }

            shortArmRotate = (hour * 30) + (minute * 0.5);
            longArmRotate = (minute * 6) + (second * 0.1);

            shortArm.css('transform', 'rotate(' + shortArmRotate + 'deg)');
            longArm.css('transform', 'rotate(' + longArmRotate + 'deg)');

            // handle alarm settings

            if (isAlarmSet) {
                if (alarmDate) {
                    var dateDifference = Math.floor((currentDate.getTime() - alarmDate.getTime()) / (1000));
                    if (dateDifference > resetAfter) {
                        if (hasGoneOff)  {
                        // when 5 minutes passed after go-off time, then reset alarm setting.
                            basePlate.attr('src', '../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Back.svg');
                            hasGoneOff = false;
                            isAlarmSet = false;
                            this.setProperty('alarm', false);
                        }
                    } else if (dateDifference > 0) {
                        // fire event.
                        if (! hasGoneOff) {
                            this.jqElement.triggerHandler('Alarmed');
                            TW.log.debug("Alarm went off at " + currentDate);
                            basePlate.attr('src', '../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_BackAlert.svg');
                            hasGoneOff = true;
                        } else {
                            TW.log.debug("Time to reset: " + (resetAfter - dateDifference));
                        }
                    } else {
                        TW.log.debug("Time to go off: " + dateDifference);                      
                    }
                }
            }
        };
    };
};

とくに難しいところはないのですが、順番に見ていきましょう。

冒頭部分で、アラーム機能の実装に必要ないくつかのローカル変数の定義を追加しています。

analog_clock.runtime.js
    var basePlate;
    var shortArmRotate;
    var longArmRotate;
    var currentDate;
    var alarmDate;
    var isAlarmSet;
    var hasGoneOff;
    var resetAfter;

つぎに、renderHtml()関数を見てみます。

analog_clock.runtime.js
    this.renderHtml = function () {
        return  '<div class="widget-content widget-analog_clock">' +
                    '<img class="aClock aClock_base"     src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Back.svg" />' +
                    '<img class="aClock aClock_shortArm" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_ShortArm.svg" />' +
                    '<img class="aClock aClock_longArm"  src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_LongArm.svg" />' +
                    '<img class="aClock" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Center.svg" />' +
                '</div>';
    };

小さな違いですが、文字盤の画像を読み込んでいる img タグに aClock_baseクラスを新たに追加しています。これは、後ほど「アラーム状態になったときに文字盤を赤くする」という処理のために必要です。

つづいてafterRender()関数ですね。

analog_clock.runtime.js
    this.afterRender = function () {
        basePlate = this.jqElement.find('.aClock_base');
        shortArm = this.jqElement.find('.aClock_shortArm');
        longArm = this.jqElement.find('.aClock_longArm');
        alarmDate = this.getProperty('alarm date');
        isAlarmSet = this.getProperty('alarm');
        hasGoneOff = false;
        resetAfter = this.getProperty('reset after');
    };

冒頭に追加したいくつかのローカル変数に値をセットしています。this.getProperty()関数は、マッシュアップビルダーで指定したプロパティを読み込むために使います。

どんどんいきましょう。今回の改造の山場、updateProperties()関数です。ちょっと長いですが、大丈夫。難しい処理はしていません。

analog_clock.runtime.js
        if (updatePropertyInfo.TargetProperty === 'alarm') {
            if (updatePropertyInfo.SinglePropertyValue === 'false') {
                isAlarmSet = false;
            } else {
                isAlarmSet = true;
            }
            this.setProperty('alarm', isAlarmSet);
        }

        if (updatePropertyInfo.TargetProperty === 'alarm date') {
            alarmDate = new Date(updatePropertyInfo.SinglePropertyValue);
            this.setProperty('alarm date', alarmDate);
        }

        if (updatePropertyInfo.TargetProperty === 'reset after') {
            resetAfter = parseInt(updatePropertyInfo.SinglePropertyValue, 10);
            this.setProperty('reset after', resetAfter);
        }

まず、関数の最初に alart プロパティや alart date プロパティが変更された際の処理を書いています。setProperty()関数は、JavaScript コードの内部からウィジェットのプロパティに値を書き込むために使います。ユーザーやプログラム処理によってウィジェットのプロパティを変更した際は、とくにそのプロパティがバインドソースとして定義されている場合には、忘れずにsetProperty()を呼ぶようにしましょう。

つづいて、date プロパティが更新された時の処理ですが、後半部分にアラーム処理を書き加えています。下記の部分ですね。

analog_clock.runtime.js
            // handle alarm settings

            if (isAlarmSet) {
                if (alarmDate) {
                    var dateDifference = Math.floor((currentDate.getTime() - alarmDate.getTime()) / (1000));
                    if (dateDifference > resetAfter) {
                        if (hasGoneOff)  {
                        // when 5 minutes passed after go-off time, then reset alarm setting.
                            basePlate.attr('src', '../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Back.svg');
                            hasGoneOff = false;
                            isAlarmSet = false;
                            this.setProperty('alarm', false);
                        }
                    } else if (dateDifference > 0) {
                        // fire event.
                        if (! hasGoneOff) {
                            this.jqElement.triggerHandler('Alarmed');
                            TW.log.debug("Alarm went off at " + currentDate);
                            basePlate.attr('src', '../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_BackAlert.svg');
                            hasGoneOff = true;
                        } else {
                            TW.log.debug("Time to reset: " + (resetAfter - dateDifference));
                        }
                    } else {
                        TW.log.debug("Time to go off: " + dateDifference);                      
                    }
                }
            }

かなり深いif文になっていますが、アラーム状態か否かを判断して、アラーム状態であれば下記の処理を実施しています。

  • aClock_baseクラスの要素の画像を差し替える(赤い文字盤の画像が指定されています)
  • alarmedイベントを発行する

また、アラーム状態になってから reset after 秒以上経過すると、アラーム状態を解除するために次の処理をしています。

  • aClock_baseクラスの要素の画像を通常のものに差し替える

ほぼ通常の JavaScript か JQuery のシンタックスですので難しいところはないかと思います。ThingWorx の香りがする部分は下記ですね。

analog_clock.runtime.js
                            this.jqElement.triggerHandler('Alarmed');
                            TW.log.debug("Alarm went off at " + currentDate);

jqElement.triggerhandler('alarmed')は JQuery のシンタックスですが、この一行でウィジェットに定義されているalarmedイベントを発行しています。このalarmedという名前は、analog_clock.ide.js のthis.wigetEvents()関数で定義したものと名前が一致する必要があるので、注意してください。

つづくTW.log.debug()は、ログ出力のためです。このログは、ブラウザのコンソールへ送られます。

最後の一歩:ウィジェットを使ってみる

以上でアラーム機能の作り込みは終わりました。metadata.xml の packageVersion番号をあげて、エクステンションをビルドして ThingWorx にインポートすれば作業は終了です。

せっかく新しい機能を作ったので、実際に動作するかマッシュアップ上でテストしてみましょう。以下のように前回作った "MyClock"マッシュアップに変更を加えます。

  1. Date Time Picker ウィジェットを ふたつ 配置する。
  2. 新しい Expression を "WentOffAt" という名前で作る。Expression には new Date()を指定し、Output には DATETIME 型を指定する。
  3. Check Box ウィジェットをひとつマッシュアップに配置する。Prompt プロパティには "Set Alarm" と入力する。

マッシュアップの登場人物は以上です。マッシュアップビルダー場では次のようにレイアウトエリアに表示されていると思います。

image.png

上記画像では追加で Label ウィジェットも使っていますが、動作に影響しないものなので必須ではありません。Function ペインは次のようになっています。

image.png

次に、配置したウィジェットをバインドしていきます。最初にバインドした結果の画像をみてみましょう。

image.png

  1. さきほど配置したひとつめの Date Time Picker ウィジェットの "Date Time" プロパティを Analog Clock ウィジェットの "alarm date" プロパティにバインドします。これで Date Time Picker で選択した日時が Analog Clock のアラーム時間としてセットされます。
  2. おなじく、Check Box ウィジェットの "State" プロパティを Analog Clock ウィジェットの "alarm" プロパティにバインドします。これはアラーム時計のアラームスイッチに相当します。Check Box ウィジェットにチェックが入っていればアラームが鳴り、そうでなければアラーム時刻をセットしていてもアラームは鳴りません。
  3. Analog Clock ウィジェットの "Alarmed" イベントを WentOffAt Expression の "Evaluate" 入力パラメタにバインドします。こうすることで、Analog Clock がアラート状態になった時に、WengOffAt Expression が呼び出されます。

あと一息です。

image.png

  1. WentOffAt Expression の "Output" を、先ほど追加した二つ目の Date Time Picker ウィジェットの "Date Time" プロパティにバインドしてください。この操作で、Analog Clock がアラート状態になった時間(Alarmed イベントが発行された時間)を表示します。

すべてのバインドがうまくいったら、マッシュアップを保存して "View Mashup" ボタンをクリックしてマッシュアップを実行してください。次のような表示になります。

image.png

一つ目の Date Time Picker (上図では "Go-off Time")で時間を指定し、"Set Alarm" をチェックすると、Analog Clock にアラームが設定されます。

アラームに設定した時間に到達すると、下記のような表示になります。

image.png

文字盤が赤くなり、さらに二つ目の Date Time Picker の表示が "Select Date/Time" からアラームが鳴った時間に変わっていることがわかります。このまま実行を続けると、Analog Clock ウィジェットの "reset After" プロパティで指定した時間が経過すると、文字盤が元に戻ります。

image.png

おわりに

これまでの記事で、以下を説明しました。

  • ウィジェットをつくるために Eclpise 用のプラグインを入れる方法
  • 簡単なウィジェットを作る方法
  • プロパティを受け取る方法(時刻情報を受け取りました)
  • 入力に応じて表現を変える方法(CSSのtransformを使用しました)
  • イベントの発行のしかた(Alarmedイベントを発行しました)

ウィジェットにはこのほか、独自のサービスを提供することもできますが、まずはここまでで紹介した機能である程度「使える」ウィジェットは作れるのではないかと思います。

第一回でも説明しましたが、詳しい情報は下記の URL にあります。残念ながら英語ですが、必要な情報はほぼ網羅されていますので、ご参考までに。

ThingWorx Extension Development Guide

余談

ところで、Expression の名前に "WentOffAt" という名前を使いました。また、JavaScript コードの中でhasGoneOffというローカル変数を使っています。どちらも原型は "go off" となりますが、この "go off" とは何でしょうか?

グーグル先生の教えてくれるところによると、アラームが「鳴る」ことを英語では "go off" と表現するんだそうです。へぇぇ...。

今回作成したリソース全掲

metadata.xml
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<Entities>
    <ExtensionPackages>
        <ExtensionPackage dependsOn="" description="" minimumThingWorxVersion="8.3.0" name="AnalogClock" packageVersion="1.4.1" vendor="Kanoloa"> 
        </ExtensionPackage>
    </ExtensionPackages>
    <Widgets>
        <Widget name="analog_clock">
            <UIResources>
                <FileResource description="" file="analog_clock.ide.js" isDevelopment="true" isRuntime="false" type="JS"/>
                <FileResource description="" file="analog_clock.ide.css" isDevelopment="true" isRuntime="false" type="CSS"/>
                <FileResource description="" file="analog_clock.runtime.js" isDevelopment="false" isRuntime="true" type="JS"/>
                <FileResource description="" file="analog_clock.runtime.css" isDevelopment="false" isRuntime="true" type="CSS"/>
            </UIResources>
        </Widget>
    </Widgets>
</Entities>
analog_clock.ide.js
TW.IDE.Widgets.analog_clock = function () {

    this.widgetIconUrl = function() {
        return  "'../Common/extensions/AnalogClock/ui/analog_clock/default_widget_icon.ide.png'";
    };

    this.widgetProperties = function () {
        return {
            'name': 'Analog Clock',
            'description': '',
            'category': ['Common'],
            'isResizable': true,
            'supportsAutoResize': true,
            'properties': {
                'date' : {
                    'baseType': 'DATETIME',
                    'isBindingTarget' : true,
                    'isVisible': true
                },
                'alarm date': {
                    'baseType': 'DATETIME',
                    'isBindingTarget': true,
                    'isVisible': true
                },
                'alarm': {
                    'baseType': 'BOOLEAN',
                    'defaultValue': false,
                    'isEditable': true,
                    'isBindingTarget': true,
                    'isVisible': true
                },
                'reset after': {
                    'baseType': 'INTEGER',
                    'defaultValue': 300,
                    'isEditable': true,
                    'isBindingTarget': true,
                    'isVisible': true
                },
                'Width': {
                    'description': TW.IDE.I18NController.translate('tw.textbox-ide.properties.width.description'),
                    'defaultValue': 120,
                    'isEditable': true,
                    'isVisible': true
                },
                'Height': {
                    'description': TW.IDE.I18NController.translate('tw.textbox-ide.properties.height.description'),
                    'defaultValue': 120,
                    'isEditable': true,
                    'isVisible': true
                }
            }
        }
    };

    this.afterSetProperty = function (name, value) {
        var thisWidget = this;
        var refreshHtml = false;
        switch (name) {
            case 'Style':
            case 'Width':
                this.setProperty('Height', value);
            case 'Height':
                this.setProperty('Width', value);
            case 'Alignment':
                refreshHtml = true;
                break;
            default:
                break;
        }
        return refreshHtml;
    };

    this.renderHtml = function () {
        // return any HTML you want rendered for your widget
        // If you want it to change depending on properties that the user
        // has set, you can use this.getProperty(propertyName).
        return  '<div class="widget-content widget-analog_clock">' +
                    '<img class="aClock" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Back.svg"> ' +
                    '<img class="aClock aClock_shortArm" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_ShortArm.svg" />' +
                    '<img class="aClock aClock_longArm" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_LongArm.svg" />' +
                    '<img class="aClock" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Center.svg" />' +
                '</div>';
    };

    this.afterRender = function () {
        // NOTE: this.jqElement is the jquery reference to your html dom element
        //       that was returned in renderHtml()
    };

    this.widgetEvents = function() {
        return {
            'Alarmed': {}
        };
    };


};
analog_clock.ide.css
/* Place custom CSS styling for Analog Clock widget in Composer in this file */

.widget-analog_clock {
    position: relative;
}

.aClock {
    position: absolute;
    left: 0;
    right: 0;
    max-width: 100%;
    height: auto;
    transition: 0.5s;
}
analog_clock.runtime.js
TW.Runtime.Widgets.analog_clock= function () {

    // var thisWidget = this;
    var shortArm;
    var longArm;
    var hour;
    var minute;
    var second;
    var basePlate;
    var shortArmRotate;
    var longArmRotate;
    var currentDate;
    var alarmDate;
    var isAlarmSet;
    var hasGoneOff;
    var resetAfter;

    this.renderHtml = function () {
        return  '<div class="widget-content widget-analog_clock">' +
                    '<img class="aClock aClock_base"     src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Back.svg" />' +
                    '<img class="aClock aClock_shortArm" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_ShortArm.svg" />' +
                    '<img class="aClock aClock_longArm"  src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_LongArm.svg" />' +
                    '<img class="aClock" src="../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Center.svg" />' +
                '</div>';
    };

    this.afterRender = function () {
        basePlate = this.jqElement.find('.aClock_base');
        shortArm = this.jqElement.find('.aClock_shortArm');
        longArm = this.jqElement.find('.aClock_longArm');
        alarmDate = this.getProperty('alarm date');
        isAlarmSet = this.getProperty('alarm');
        hasGoneOff = false;
        resetAfter = this.getProperty('reset after');
    };

    this.updateProperty = function (updatePropertyInfo) {

        if (updatePropertyInfo.TargetProperty === 'alarm') {
            if (updatePropertyInfo.SinglePropertyValue === 'false') {
                isAlarmSet = false;
            } else {
                isAlarmSet = true;
            }
            this.setProperty('alarm', isAlarmSet);
        }

        if (updatePropertyInfo.TargetProperty === 'alarm date') {
            alarmDate = new Date(updatePropertyInfo.SinglePropertyValue);
            this.setProperty('alarm date', alarmDate);
        }

        if (updatePropertyInfo.TargetProperty === 'reset after') {
            resetAfter = parseInt(updatePropertyInfo.SinglePropertyValue, 10);
            this.setProperty('reset after', resetAfter);
        }

        if (updatePropertyInfo.TargetProperty === 'date') {

            currentDate = new Date(updatePropertyInfo.SinglePropertyValue);
            hour = currentDate.getHours();
            minute = currentDate.getMinutes();
            second = currentDate.getSeconds();

            if (hour > 11) {
                hour = hour - 12;
            } else if (!hour) {
                hour = 0;
            }

            if (minute === '60') {
                minute = 0;
            } else if (!minute) {
                minute = 0;
            }

            shortArmRotate = (hour * 30) + (minute * 0.5);
            longArmRotate = (minute * 6) + (second * 0.1);

            shortArm.css('transform', 'rotate(' + shortArmRotate + 'deg)');
            longArm.css('transform', 'rotate(' + longArmRotate + 'deg)');

            // handle alarm settings

            if (isAlarmSet) {
                if (alarmDate) {
                    var dateDifference = Math.floor((currentDate.getTime() - alarmDate.getTime()) / (1000));
                    if (dateDifference > resetAfter) {
                        if (hasGoneOff)  {
                        // when 5 minutes passed after go-off time, then reset alarm setting.
                            basePlate.attr('src', '../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_Back.svg');
                            hasGoneOff = false;
                            isAlarmSet = false;
                            this.setProperty('alarm', false);
                        }
                    } else if (dateDifference > 0) {
                        // fire event.
                        if (! hasGoneOff) {
                            this.jqElement.triggerHandler('Alarmed');
                            TW.log.debug("Alarm went off at " + currentDate);
                            basePlate.attr('src', '../Common/extensions/AnalogClock/ui/analog_clock/image/AnalogClock_BackAlert.svg');
                            hasGoneOff = true;
                        } else {
                            TW.log.debug("Time to reset: " + (resetAfter - dateDifference));
                        }
                    } else {
                        TW.log.debug("Time to go off: " + dateDifference);                      
                    }
                }
            }
        };
    };
};
analog_clock.runtime.css
/* Place custom CSS styling for Analog Clock widget at runtime in this file */

.widget-analog_clock {
    position: relative;
}

.aClock {
    position: absolute;
    left: 0;
    right: 0;
    max-width: 100%;
    height: auto;
    transition: 0.5s;
}

リソース

ソースコードや画像は下記から実動するものをダウンロードできます。

Why not register and get more from Qiita?
  1. We will deliver articles that match you
    By following users and tags, you can catch up information on technical fields that you are interested in as a whole
  2. you can read useful information later efficiently
    By "stocking" the articles you like, you can search right away
Comments
No comments
Sign up for free and join this conversation.
If you already have a Qiita account
Why do not you register as a user and use Qiita more conveniently?
You need to log in to use this function. Qiita can be used more conveniently after logging in.
You seem to be reading articles frequently this month. Qiita can be used more conveniently after logging in.
  1. We will deliver articles that match you
    By following users and tags, you can catch up information on technical fields that you are interested in as a whole
  2. you can read useful information later efficiently
    By "stocking" the articles you like, you can search right away
ユーザーは見つかりませんでした