始める前に
この記事では、プレイヤーの頭を使用して Minecraft: Java Edition で現実世界の日付と時刻を取得する関数を作成します。取得した結果は整数としてストレージに保存されます。
検証済みバージョンは 1.21.4 です。
前提条件
- 基本的な数学の知識を理解していること。
-
data
、scoreboard
、マクロなど、コマンドおよび関数に精通していること。 - 「Base64 形式にエンコードされた文字列」をテキストにデコードする方法を熟知していること。
プレイヤーの頭から UNIX 時刻を取得する
コマンド give @s minecraft:player_head[minecraft:profile= "MHF_Question"] 1
を使用して「プレイヤーの頭」を入手すると、次のようなアイテム データが保存されます。
このアイテム データの value
には、スキンのテクスチャ データが「Base64 形式にエンコードされた文字列」として保存されます。
この「エンコードされた文字列」をデコードすると、次の文字列を取得できます。
timestamp
には、「テクスチャ データを取得した日時の UNIX 時刻」が保存されます。
UNIX 時刻とは、1970年1月1日 00:00:00(UTC)からの経過秒数です。つまり、この値を取得してしまえば、現実世界の日時を算出することができます。
プレイヤーの頭から「エンコードされた文字列」を取得する
プレイヤーの頭をアイテムとして召喚して、「エンコードされた文字列」を取得します。
まずはプレイヤーの頭をアイテムとして召喚します。
bring.mcfunction
# イニシャライズする
data remove storage current_time_bringer:current_time current_time
data remove storage current_time_bringer:current_time data
data modify storage current_time_bringer:current_time data.binary set value ""
data modify storage current_time_bringer:current_time data.unix set value ""
# アイテムを召喚する
summon minecraft:item ~ ~ ~ {\
Item: {count: 1, components: { "minecraft:profile": {"name": "MHF_Question"}}, id: "minecraft:player_head"},\
PickupDelay: 32767, Tags: ["tag.current_time_bringer.item"]\
}
# 次の処理を開始する
function current_time_bringer:fetch_unix/repeat_detection
「エンコードされた文字列」が保存されるのは約 1 秒後です。そのため、「エンコードされた文字列」が保存されるまで再帰処理を行う必要があります。
「エンコードされた文字列」が保存されていない場合は、components."minecraft:profile".id
というデータがありません。そのデータがない場合は 1 ティック後に同じ関数を実行し、そのデータがある場合は次の処理を開始します。
fetch_unix/repeat_detection.mcfunction
# 「プレイヤーの頭の情報」が保存されている場合は次の処理を開始する
execute if entity @n[tag= tag.current_time_bringer.item] if data entity @n[tag= tag.current_time_bringer.item] Item.components."minecraft:profile".id if data entity @n[tag= tag.current_time_bringer.item] Item.components."minecraft:profile".properties[].value run function current_time_bringer:fetch_unix/run
# 「プレイヤーの頭の情報」が保存されていない場合は 1 ティック後に同じ関数を実行する
execute if entity @n[tag= tag.current_time_bringer.item] unless data entity @n[tag= tag.current_time_bringer.item] Item.components."minecraft:profile".id run schedule function current_time_bringer:fetch_unix/repeat_detection 1t append
次に、アイテムから「エンコードされた文字列」を 14 文字に切り取って取得し、アイテムをキルします。
fetch_unix/run.mcfunction
# 「エンコードされた文字列」を 14 文字に切り取る
data modify storage current_time_bringer:current_time data.raw set from entity @n[tag= tag.current_time_bringer.item] Item.components."minecraft:profile".properties[].value
data modify storage current_time_bringer:current_time data.raw set string storage current_time_bringer:current_time data.raw 24 38
# アイテムをキルする
kill @e[tag= tag.current_time_bringer.item]
# 次の処理を開始する
function current_time_bringer:fetch_unix/repeat_raw_separation
ここで「エンコードされた文字列」を 14 文字に切り取っているのは、こっちの方が効率が良さそうだと思ったからです。
「エンコードされた文字列」をデコードする
取得した「エンコードされた文字列」をデコードします。
まずは、Base64 アルファベットに基づいて、文字をバイナリに変換します。
fetch_unix/repeat_raw_separation.mcfunction
# 「エンコードされた文字列」から 1 文字を切り離す
data modify storage current_time_bringer:current_time data.separated_raw set string storage current_time_bringer:current_time data.raw 0 1
data modify storage current_time_bringer:current_time data.raw set string storage current_time_bringer:current_time data.raw 1
# 切り離した文字をバイナリに変換する関数を実行する
function current_time_bringer:fetch_unix/convert_to_binary
# バイナリを結合する関数を実行する
function current_time_bringer:fetch_unix/concatenate_strings_to_binary with storage current_time_bringer:current_time data
# 「エンコードされた文字列」が残っていない場合は次の処理を開始する
execute if data storage current_time_bringer:current_time data{raw: ""} run function current_time_bringer:fetch_unix/process_before_converting_to_ascii
# 「エンコードされた文字列」が残っている場合は同じ関数を繰り返す
execute unless data storage current_time_bringer:current_time data{raw: ""} if data storage current_time_bringer:current_time data.raw run function current_time_bringer:fetch_unix/repeat_raw_separation
fetch_unix/convert_to_binary.mcfunction
# 切り離した文字が「A」の場合はバイナリ「000000」を返戻する
execute if data storage current_time_bringer:current_time data{separated_raw: "A"} run data modify storage current_time_bringer:current_time data.string_for_appending set value "000000"
# 切り離した文字が「B」の場合はバイナリ「000001」を返戻する
execute if data storage current_time_bringer:current_time data{separated_raw: "B"} run data modify storage current_time_bringer:current_time data.string_for_appending set value "000001"
...
# 切り離した文字が「/」の場合はバイナリ「111111」を返戻する
execute if data storage current_time_bringer:current_time data{separated_raw: "/"} run data modify storage current_time_bringer:current_time data.string_for_appending set value "111111"
# 切り離した文字が「=」の場合はバイナリを返戻しない
execute if data storage current_time_bringer:current_time data{separated_raw: "="} run data modify storage current_time_bringer:current_time data.string_for_appending set value ""
data remove storage current_time_bringer:current_time data.separated_raw
fetch_unix/concatenate_strings_to_binary.mcfunction
# バイナリを結合する
$data modify storage current_time_bringer:current_time data.binary set value "$(binary)$(string_for_appending)"
data remove storage current_time_bringer:current_time data.string_for_appending
fetch_unix/process_before_converting_to_ascii.mcfunction
data remove storage current_time_bringer:current_time data.raw
# 次の処理を開始する
function current_time_bringer:fetch_unix/repeat_binary_separation
次に、バイナリを ASCII に変換します。
fetch_unix/repeat_binary_separation.mcfunction
# バイナリから 8 文字を切り離す
data modify storage current_time_bringer:current_time data.separated_binary set string storage current_time_bringer:current_time data.binary 0 8
data modify storage current_time_bringer:current_time data.binary set string storage current_time_bringer:current_time data.binary 4
data modify storage current_time_bringer:current_time data.binary set string storage current_time_bringer:current_time data.binary 4
# バイナリを ASCII に変換する関数を実行する
function current_time_bringer:fetch_unix/convert_to_ascii
# ASCII を結合する関数を実行する
function current_time_bringer:fetch_unix/concatenate_strings_to_unix with storage current_time_bringer:current_time data
# バイナリが残っていない場合は次の処理を開始する
execute if data storage current_time_bringer:current_time data{binary: ""} run function current_time_bringer:fetch_unix/process_before_converting_to_date_time_components
# バイナリが残っている場合は同じ関数を繰り返す
execute unless data storage current_time_bringer:current_time data{binary: ""} if data storage current_time_bringer:current_time data.binary run function current_time_bringer:fetch_unix/repeat_binary_separation
fetch_unix/convert_to_ascii.mcfunction
# バイナリが「00101101」の場合は ASCII「-」を返戻する
execute if data storage current_time_bringer:current_time data{separated_binary: "00101101"} run data modify storage current_time_bringer:current_time data.string_for_appending set value "-"
# バイナリが「00110000」の場合は ASCII「0」を返戻する
execute if data storage current_time_bringer:current_time data{separated_binary: "00110000"} run data modify storage current_time_bringer:current_time data.string_for_appending set value "0"
...
# バイナリが「00111000」の場合は ASCII「8」を返戻する
execute if data storage current_time_bringer:current_time data{separated_binary: "00111000"} run data modify storage current_time_bringer:current_time data.string_for_appending set value "8"
# バイナリが「00111001」の場合は ASCII「9」を返戻する
execute if data storage current_time_bringer:current_time data{separated_binary: "00111001"} run data modify storage current_time_bringer:current_time data.string_for_appending set value "9"
data remove storage current_time_bringer:current_time data.separated_binary
fetch_unix/concatenate_strings_to_unix.mcfunction
# ASCII を結合する
$data modify storage current_time_bringer:current_time data.unix set value "$(unix)$(string_for_appending)"
data remove storage current_time_bringer:current_time data.string_for_appending
fetch_unix/process_before_converting_to_date_time_components.mcfunction
data remove storage current_time_bringer:current_time data.binary
# スコアボード オブジェクトを作成する
scoreboard objectives add objective.current_time_bringer.temporary_value dummy {"fallback": "Temporary Value", "translate": "objective.current_time_bringer.temporary_value", "type": "translatable"}
# 次の処理を開始する
function current_time_bringer:convert_to_date_time_components/tick with storage current_time_bringer:current_time data
# スコアボード オブジェクトを削除する
scoreboard objectives remove objective.current_time_bringer.temporary_value
UNIX 時刻が文字列としてストレージ current_time_bringer:current_time
の data.unix
に保存されます。
UNIX 時刻を日時の構成要素に変換する
UNIX 時刻を日時の構成要素に変換するには、次のプロセスを使用します。
convert_to_date_time_components/tick.mcfunction
# UNIX 時刻を代入する
$scoreboard players set target.current_time_bringer.unix objective.current_time_bringer.temporary_value $(unix)
# 変数を定義する
function current_time_bringer:convert_to_date_time_components/define/days_in_month
scoreboard players set target.current_time_bringer.seconds_in_day objective.current_time_bringer.temporary_value 86400
# UNIX 時刻から日数と残りの秒数を算出する関数を実行する
function current_time_bringer:convert_to_date_time_components/define/days_and_remaining_seconds
# 日数から現在の年を算出する
scoreboard players set target.current_time_bringer.year objective.current_time_bringer.temporary_value 1970
scoreboard players set target.current_time_bringer.days_in_year objective.current_time_bringer.temporary_value 365
scoreboard players set target.current_time_bringer.leap_year_days objective.current_time_bringer.temporary_value 366
execute if score target.current_time_bringer.days objective.current_time_bringer.temporary_value >= target.current_time_bringer.days_in_year objective.current_time_bringer.temporary_value run function current_time_bringer:convert_to_date_time_components/convert_days_to_year_and_month/tick
# 月の日数を考慮して現在の月を算出する
scoreboard players set target.current_time_bringer.month objective.current_time_bringer.temporary_value 1
function current_time_bringer:convert_to_date_time_components/define/is_leap_year
scoreboard players operation target.current_time_bringer.days_in_month.2 objective.current_time_bringer.temporary_value += target.current_time_bringer.is_leap_year objective.current_time_bringer.temporary_value
execute store result storage current_time_bringer:current_time convert_to_date_time_components.month int 1 run scoreboard players get target.current_time_bringer.month objective.current_time_bringer.temporary_value
function current_time_bringer:convert_to_date_time_components/calculate_taking_into_account_days_in_month/tick with storage current_time_bringer:current_time convert_to_date_time_components
data remove storage current_time_bringer:current_time convert_to_date_time_components
# 現在の日を算出する
scoreboard players operation target.current_time_bringer.day objective.current_time_bringer.temporary_value = target.current_time_bringer.days objective.current_time_bringer.temporary_value
scoreboard players add target.current_time_bringer.day objective.current_time_bringer.temporary_value 1
# 残りの秒数から現在の時、分、秒を算出する
scoreboard players set target.current_time_bringer.seconds_in_hour objective.current_time_bringer.temporary_value 3600
scoreboard players operation target.current_time_bringer.hour objective.current_time_bringer.temporary_value = target.current_time_bringer.remaining_seconds objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.hour objective.current_time_bringer.temporary_value /= target.current_time_bringer.seconds_in_hour objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.remaining_seconds objective.current_time_bringer.temporary_value %= target.current_time_bringer.seconds_in_hour objective.current_time_bringer.temporary_value
scoreboard players set target.current_time_bringer.seconds_in_minute objective.current_time_bringer.temporary_value 60
scoreboard players operation target.current_time_bringer.minute objective.current_time_bringer.temporary_value = target.current_time_bringer.remaining_seconds objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.minute objective.current_time_bringer.temporary_value /= target.current_time_bringer.seconds_in_minute objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.second objective.current_time_bringer.temporary_value = target.current_time_bringer.remaining_seconds objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.second objective.current_time_bringer.temporary_value %= target.current_time_bringer.seconds_in_minute objective.current_time_bringer.temporary_value
# 算出した現在の年、月、日、時、分、秒をストレージに保存する
execute store result storage current_time_bringer:current_time current_time.year int 1 run scoreboard players get target.current_time_bringer.year objective.current_time_bringer.temporary_value
execute store result storage current_time_bringer:current_time current_time.month int 1 run scoreboard players get target.current_time_bringer.month objective.current_time_bringer.temporary_value
execute store result storage current_time_bringer:current_time current_time.day int 1 run scoreboard players get target.current_time_bringer.day objective.current_time_bringer.temporary_value
execute store result storage current_time_bringer:current_time current_time.hour int 1 run scoreboard players get target.current_time_bringer.hour objective.current_time_bringer.temporary_value
execute store result storage current_time_bringer:current_time current_time.minute int 1 run scoreboard players get target.current_time_bringer.minute objective.current_time_bringer.temporary_value
execute store result storage current_time_bringer:current_time current_time.second int 1 run scoreboard players get target.current_time_bringer.second objective.current_time_bringer.temporary_value
data remove storage current_time_bringer:current_time data
convert_to_date_time_components/define/days_in_month.mcfunction
# 1 月の日数を定義する
scoreboard players set target.current_time_bringer.days_in_month.1 objective.current_time_bringer.temporary_value 31
# 2 月の日数を定義する
scoreboard players set target.current_time_bringer.days_in_month.2 objective.current_time_bringer.temporary_value 28
...
# 11 月の日数を定義する
scoreboard players set target.current_time_bringer.days_in_month.11 objective.current_time_bringer.temporary_value 30
# 12 月の日数を定義する
scoreboard players set target.current_time_bringer.days_in_month.12 objective.current_time_bringer.temporary_value 31
convert_to_date_time_components/define/days_and_remaining_seconds.mcfunction
# 日数と残りの秒数を算出する
scoreboard players operation target.current_time_bringer.days objective.current_time_bringer.temporary_value = target.current_time_bringer.unix objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.remaining_seconds objective.current_time_bringer.temporary_value = target.current_time_bringer.unix objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.days objective.current_time_bringer.temporary_value /= target.current_time_bringer.seconds_in_day objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.remaining_seconds objective.current_time_bringer.temporary_value %= target.current_time_bringer.seconds_in_day objective.current_time_bringer.temporary_value
convert_to_date_time_components/define/is_leap_year.mcfunction
# うるう年であるか確認する
scoreboard players set target.current_time_bringer.is_leap_year objective.current_time_bringer.temporary_value 0
scoreboard players operation target.current_time_bringer.year_00 objective.current_time_bringer.temporary_value = target.current_time_bringer.year objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.year_01 objective.current_time_bringer.temporary_value = target.current_time_bringer.year objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.year_02 objective.current_time_bringer.temporary_value = target.current_time_bringer.year objective.current_time_bringer.temporary_value
scoreboard players set target.current_time_bringer.4 objective.current_time_bringer.temporary_value 4
scoreboard players set target.current_time_bringer.100 objective.current_time_bringer.temporary_value 100
scoreboard players set target.current_time_bringer.400 objective.current_time_bringer.temporary_value 400
scoreboard players operation target.current_time_bringer.year_00 objective.current_time_bringer.temporary_value %= target.current_time_bringer.4 objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.year_01 objective.current_time_bringer.temporary_value %= target.current_time_bringer.100 objective.current_time_bringer.temporary_value
scoreboard players operation target.current_time_bringer.year_02 objective.current_time_bringer.temporary_value %= target.current_time_bringer.400 objective.current_time_bringer.temporary_value
execute if score target.current_time_bringer.year_00 objective.current_time_bringer.temporary_value matches 0 unless score target.current_time_bringer.year_01 objective.current_time_bringer.temporary_value matches 0 run scoreboard players set target.current_time_bringer.is_leap_year objective.current_time_bringer.temporary_value 1
execute unless score target.current_time_bringer.is_leap_year objective.current_time_bringer.temporary_value matches 1 if score target.current_time_bringer.year_02 objective.current_time_bringer.temporary_value matches 0 run scoreboard players set target.current_time_bringer.is_leap_year objective.current_time_bringer.temporary_value 1
convert_to_date_time_components/convert_days_to_year_and_month/tick.mcfunction
# うるう年であるか確認する
function current_time_bringer:convert_to_date_time_components/define/is_leap_year
# うるう年であり日数が 366 以上である場合は次の関数を実行する
execute if score target.current_time_bringer.is_leap_year objective.current_time_bringer.temporary_value matches 1 if score target.current_time_bringer.days objective.current_time_bringer.temporary_value >= target.current_time_bringer.leap_year_days objective.current_time_bringer.temporary_value run function current_time_bringer:convert_to_date_time_components/convert_days_to_year_and_month/is_leap_year
# うるう年であり日数が 366 以上ではない場合はループを終了する
execute if score target.current_time_bringer.is_leap_year objective.current_time_bringer.temporary_value matches 1 unless score target.current_time_bringer.days objective.current_time_bringer.temporary_value >= target.current_time_bringer.leap_year_days objective.current_time_bringer.temporary_value run scoreboard players set target.current_time_bringer.stop_loop objective.current_time_bringer.temporary_value 1
# うるう年ではない場合は次の関数を実行する
execute unless score target.current_time_bringer.is_leap_year objective.current_time_bringer.temporary_value matches 1 run function current_time_bringer:convert_to_date_time_components/convert_days_to_year_and_month/is_not_leap_year
# 日数が 365 以上である場合は同じ関数を繰り返す
execute unless score target.current_time_bringer.stop_loop objective.current_time_bringer.temporary_value matches 1 if score target.current_time_bringer.days objective.current_time_bringer.temporary_value >= target.current_time_bringer.days_in_year objective.current_time_bringer.temporary_value run function current_time_bringer:convert_to_date_time_components/convert_days_to_year_and_month/tick
convert_to_date_time_components/convert_days_to_year_and_month/is_leap_year.mcfunction
# 日数から 366 を減算する
scoreboard players remove target.current_time_bringer.days objective.current_time_bringer.temporary_value 366
# 年に 1 を加算する
scoreboard players add target.current_time_bringer.year objective.current_time_bringer.temporary_value 1
convert_to_date_time_components/convert_days_to_year_and_month/is_not_leap_year.mcfunction
# 日数から 365 を減算する
scoreboard players remove target.current_time_bringer.days objective.current_time_bringer.temporary_value 365
# 年に 1 を加算する
scoreboard players add target.current_time_bringer.year objective.current_time_bringer.temporary_value 1
convert_to_date_time_components/calculate_taking_into_account_days_in_month/tick.mcfunction
# 日数が月の日数以上である場合は次の関数を実行する
$execute if score target.current_time_bringer.days objective.current_time_bringer.temporary_value >= target.current_time_bringer.days_in_month.$(month) objective.current_time_bringer.temporary_value run function current_time_bringer:convert_to_date_time_components/calculate_taking_into_account_days_in_month/days_is_greater_than_days_in_month with storage current_time_bringer:current_time convert_to_date_time_components
convert_to_date_time_components/calculate_taking_into_account_days_in_month/days_is_greater_than_days_in_month.mcfunction
# 日数から月の日数を減算する
$scoreboard players operation target.current_time_bringer.days objective.current_time_bringer.temporary_value -= target.current_time_bringer.days_in_month.$(month) objective.current_time_bringer.temporary_value
# 月に 1 を加算する
scoreboard players add target.current_time_bringer.month objective.current_time_bringer.temporary_value 1
# 日数が月の日数以上であるか確認する
execute store result storage current_time_bringer:current_time convert_to_date_time_components.month int 1 run scoreboard players get target.current_time_bringer.month objective.current_time_bringer.temporary_value
function current_time_bringer:convert_to_date_time_components/calculate_taking_into_account_days_in_month/tick with storage current_time_bringer:current_time convert_to_date_time_components
完成したコード
この記事で完成したコードは、GitHub で確認することができます。
おわりに
プレイヤーの頭を使用して現実世界の日付と時刻を取得する関数を作成しました。
概要
- プレイヤーの頭から
components."minecraft:profile".properties[].value
を取得する。 - 取得した文字列の 25 文字目 ~ 38 文字目を Base64 形式から UNIX 時刻にデコードする。
- UNIX 時刻を日時の構成要素に変換する。
修正が必要な問題
1 つのセッションで一度しか日時を取得できない
これはゴリ押してしまえば解決できるかもしれませんが、実装が面倒だったために省略しています。
2038年1月18日 13:14:08(UTC)以降は利用できない
scoreboard
コマンドで UNIX 時刻から日時を算出していますが、スコアボード オブジェクトの最大値は 2,147,483,647 です。そのため、UNIX 時刻が 2,147,483,648-50,400=2,147,433,247 以上である 2038年1月18日 13:14:08(UTC)以降になると、正常に算出できなくなります。これはゴリ押してしまえば解決できるかもしれませんが、実装が面倒だったために省略しています。
2286年11月20日 03:46:40(UTC)以降は利用できない
「エンコードされた文字列」を 14 文字に切り取る操作は、UNIX 時刻が 10 桁であることを前提としています。そのため、UNIX 時刻が 10,000,000,000-50,400=9,999,949,600 以上である 2286年11月20日 03:46:40(UTC)以降になると、正常に算出できなくなります。そのときは私は生きていないため、生涯にわたって修正しません。