diff --git a/extensions/-SIPC-/time.js b/extensions/-SIPC-/time.js index 5690f799b9..e789b517b2 100644 --- a/extensions/-SIPC-/time.js +++ b/extensions/-SIPC-/time.js @@ -1,14 +1,30 @@ // Name: Time // ID: sipctime -// Description: Blocks for interacting with unix timestamps and other date strings. +// Description: Blocks for times, dates, and time zones. // By: -SIPC- +// By: SharkPool + +// If you're curious, the default dates are from the first commits of forkphorus & TurboWarp: +// https://github.com/forkphorus/forkphorus/commit/632d3432a8a98abd627b1309f6c85f47dcc6d428 +// https://github.com/TurboWarp/scratch-vm/commit/4a93dab4fa3704ab7a1374b9794026b3330f3433 (function (Scratch) { "use strict"; - const icon = + + const menuIconURI = ""; - const icon2 = + + const blockIconURI = ""; + + const parseDate = (str) => { + // TODO: support standalone times here, interpret as today + if (!isNaN(str)) { + return new Date(Scratch.Cast.toNumber(str)); + } + return new Date(Scratch.Cast.toString(str)); + }; + class Time { getInfo() { return { @@ -17,20 +33,18 @@ color1: "#ff8000", color2: "#804000", color3: "#804000", - menuIconURI: icon, - blockIconURI: icon2, + menuIconURI, + blockIconURI, blocks: [ { opcode: "Timestamp", blockType: Scratch.BlockType.REPORTER, text: Scratch.translate("current timestamp"), - arguments: {}, }, { opcode: "timezone", blockType: Scratch.BlockType.REPORTER, text: Scratch.translate("current time zone"), - arguments: {}, }, { opcode: "Timedata", @@ -39,23 +53,24 @@ arguments: { timestamp: { type: Scratch.ArgumentType.NUMBER, - defaultValue: "1145141980000", + defaultValue: "1591657163000", }, Timedata: { type: Scratch.ArgumentType.STRING, menu: "Time", - defaultValue: "year", }, }, }, { opcode: "TimestampToTime", blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate("convert [timestamp] to datetime"), + text: Scratch.translate( + "convert [timestamp] to YYYY-MM-DD HH:MM:SS" + ), arguments: { timestamp: { type: Scratch.ArgumentType.NUMBER, - defaultValue: "1145141980000", + defaultValue: "1591657163000", }, }, }, @@ -66,7 +81,77 @@ arguments: { time: { type: Scratch.ArgumentType.STRING, - defaultValue: "2006-04-16 06:59:40", + defaultValue: "2020-06-08 17:59:23", + }, + }, + }, + "---", + { + opcode: "differenceBetweenDateAndNow", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate( + "difference between [DATE] and now in [TIME_MENU]" + ), + arguments: { + DATE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "2020-06-08 17:59:23", + }, + TIME_MENU: { + type: Scratch.ArgumentType.STRING, + menu: "DurationUnit", + }, + }, + }, + { + opcode: "differenceBetweenDates", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate( + "difference between [START] and [END] in [TIME_MENU]" + ), + arguments: { + START: { + type: Scratch.ArgumentType.STRING, + defaultValue: "2019-01-04 18:41:04", + }, + END: { + type: Scratch.ArgumentType.STRING, + defaultValue: "2020-06-08 17:59:23", + }, + TIME_MENU: { + type: Scratch.ArgumentType.STRING, + menu: "DurationUnit", + }, + }, + }, + "---", + { + opcode: "formatTime", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("format [VALUE] seconds as [ROUND] time"), + arguments: { + VALUE: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "3883.2", // no hidden meaning in this one + }, + ROUND: { + type: Scratch.ArgumentType.NUMBER, + menu: "TimeFormat", + }, + }, + }, + { + opcode: "daysInMonth", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("number of days in [MONTH] [YEAR]"), + arguments: { + MONTH: { + type: Scratch.ArgumentType.STRING, + menu: "Months", + }, + YEAR: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "2000", }, }, }, @@ -101,18 +186,115 @@ }, ], }, + DurationUnit: { + acceptReporters: true, + items: [ + { + text: Scratch.translate("years"), + value: "years", + }, + { + text: Scratch.translate("months"), + value: "months", + }, + { + text: Scratch.translate("days"), + value: "days", + }, + { + text: Scratch.translate("hours"), + value: "hours", + }, + { + text: Scratch.translate("minutes"), + value: "minutes", + }, + { + text: Scratch.translate("seconds"), + value: "seconds", + }, + ], + }, + TimeFormat: { + acceptReporters: true, + items: [ + { + text: Scratch.translate("rounded"), + value: "rounded", + }, + { + text: Scratch.translate("exact"), + value: "exact", + }, + ], + }, + Months: { + acceptReporters: true, + items: [ + { + text: Scratch.translate("January"), + value: "1", + }, + { + text: Scratch.translate("February"), + value: "2", + }, + { + text: Scratch.translate("March"), + value: "3", + }, + { + text: Scratch.translate("April"), + value: "4", + }, + { + text: Scratch.translate("May"), + value: "5", + }, + { + text: Scratch.translate("June"), + value: "6", + }, + { + text: Scratch.translate("July"), + value: "7", + }, + { + text: Scratch.translate("August"), + value: "8", + }, + { + text: Scratch.translate("September"), + value: "9", + }, + { + text: Scratch.translate("October"), + value: "10", + }, + { + text: Scratch.translate("November"), + value: "11", + }, + { + text: Scratch.translate("December"), + value: "12", + }, + ], + }, }, }; } + Timestamp() { return Date.now(); } + timezone() { return "UTC+" + new Date().getTimezoneOffset() / -60; } + Timedata(args) { - args.timestamp = args.timestamp ? args.timestamp : null; - let date1 = new Date(Scratch.Cast.toNumber(args.timestamp)); + const date1 = parseDate(args.timestamp); switch (args.Timedata) { case "year": return date1.getFullYear(); @@ -137,31 +319,97 @@ } return 0; } + TimestampToTime({ timestamp }) { - timestamp = timestamp ? timestamp : null; - let date2 = new Date(timestamp); - let Y = date2.getFullYear() + "-"; - let M = - (date2.getMonth() + 1 < 10 - ? "0" + (date2.getMonth() + 1) - : date2.getMonth() + 1) + "-"; - let D = - (date2.getDate() < 10 ? "0" + date2.getDate() : date2.getDate()) + " "; - let h = - (date2.getHours() < 10 ? "0" + date2.getHours() : date2.getHours()) + - ":"; - let m = - (date2.getMinutes() < 10 - ? "0" + date2.getMinutes() - : date2.getMinutes()) + ":"; - let s = - date2.getSeconds() < 10 ? "0" + date2.getSeconds() : date2.getSeconds(); - return Y + M + D + h + m + s; + const date = parseDate(timestamp); + const Y = date.getFullYear(); + const M = + date.getMonth() + 1 < 10 + ? "0" + (date.getMonth() + 1) + : date.getMonth() + 1; + const D = date.getDate() < 10 ? "0" + date.getDate() : date.getDate(); + const h = date.getHours() < 10 ? "0" + date.getHours() : date.getHours(); + const m = + date.getMinutes() < 10 ? "0" + date.getMinutes() : date.getMinutes(); + const s = + date.getSeconds() < 10 ? "0" + date.getSeconds() : date.getSeconds(); + return `${Y}-${M}-${D} ${h}:${m}:${s}`; } + TimeToTimestamp({ time }) { - let data3 = time; - let timestamp = Date.parse(data3); - return timestamp; + return parseDate(time).getTime(); + } + + /** + * @param {Date} startDate + * @param {Date} endDate + * @param {string} timeMenu + * @returns {number} + */ + _calculateTimeDifference(startDate, endDate, timeMenu) { + const timeDiff = endDate.getTime() - startDate.getTime(); + switch (Scratch.Cast.toString(timeMenu)) { + case "years": + return timeDiff / (1000 * 60 * 60 * 24 * 365); + case "months": + return timeDiff / (1000 * 60 * 60 * 24 * 30.436875); // average month length from https://en.wikipedia.org/wiki/Month + case "days": + return timeDiff / (1000 * 60 * 60 * 24); + case "hours": + return timeDiff / (1000 * 60 * 60); + case "minutes": + return timeDiff / (1000 * 60); + case "seconds": + return timeDiff / 1000; + default: + return 0; + } + } + + differenceBetweenDateAndNow(args) { + return this._calculateTimeDifference( + parseDate(args.DATE), + new Date(), + args.TIME_MENU + ); + } + + differenceBetweenDates(args) { + return this._calculateTimeDifference( + parseDate(args.START), + parseDate(args.END), + args.TIME_MENU + ); + } + + formatTime(args) { + const totalSeconds = Scratch.Cast.toNumber(args.VALUE); + const seconds = + args.ROUND === "rounded" + ? Math.round(totalSeconds % 60) + .toString() + .padStart(2, "0") + : (totalSeconds % 60).toFixed(3); + const minutes = Math.round((totalSeconds / 60) % 60) + .toString() + .padStart(2, "0"); + const hours = Math.round(totalSeconds / 3600) + .toString() + .padStart(2, "0"); + return `${hours}:${minutes}:${seconds}`; + } + + daysInMonth(args) { + const year = Math.round(Scratch.Cast.toNumber(args.YEAR)); + if (year <= 0) { + return 0; + } + const monthIndex = Math.round(Scratch.Cast.toNumber(args.MONTH)); + if (monthIndex < 0 || monthIndex >= 12) { + return 0; + } + const date = new Date(year, monthIndex, 0); + return date.getDate(); } } Scratch.extensions.register(new Time());