1use std::{
2 error::Error,
3 fmt::{self, Debug},
4};
5
6use super::{private::ValueInternal, Value};
7
8use crate::{
9 context::{
10 internal::{ContextInternal, Env},
11 Context, Cx,
12 },
13 handle::{internal::TransparentNoCopyWrapper, Handle},
14 object::Object,
15 result::{JsResult, ResultExt},
16 sys::{self, raw},
17};
18
19#[cfg_attr(docsrs, doc(cfg(feature = "napi-5")))]
53#[derive(Debug)]
54#[repr(transparent)]
55pub struct JsDate(raw::Local);
56
57impl Value for JsDate {}
58
59unsafe impl TransparentNoCopyWrapper for JsDate {
60 type Inner = raw::Local;
61
62 fn into_inner(self) -> Self::Inner {
63 self.0
64 }
65}
66
67#[derive(Debug)]
69#[cfg_attr(docsrs, doc(cfg(feature = "napi-5")))]
70pub struct DateError(DateErrorKind);
71
72impl DateError {
73 pub fn kind(&self) -> DateErrorKind {
74 self.0
75 }
76}
77
78impl fmt::Display for DateError {
79 fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
80 fmt.write_str(self.0.as_str())
81 }
82}
83
84impl Error for DateError {}
85
86#[derive(Debug, Copy, Clone, PartialEq, Eq)]
88#[cfg_attr(docsrs, doc(cfg(feature = "napi-5")))]
89pub enum DateErrorKind {
90 Overflow,
93 Underflow,
96}
97
98impl DateErrorKind {
99 fn as_str(&self) -> &'static str {
100 match *self {
101 DateErrorKind::Overflow => "Date overflow",
102 DateErrorKind::Underflow => "Date underflow",
103 }
104 }
105}
106
107impl<'a, T: Value> ResultExt<Handle<'a, T>> for Result<Handle<'a, T>, DateError> {
108 fn or_throw<'b, C: Context<'b>>(self, cx: &mut C) -> JsResult<'a, T> {
110 self.or_else(|e| cx.throw_range_error(e.0.as_str()))
111 }
112}
113
114impl JsDate {
115 pub const MIN_VALUE: f64 = -8.64e15;
118 pub const MAX_VALUE: f64 = 8.64e15;
121
122 pub fn new<'a, C: Context<'a>, T: Into<f64>>(
126 cx: &mut C,
127 value: T,
128 ) -> Result<Handle<'a, JsDate>, DateError> {
129 let env = cx.env().to_raw();
130 let time = value.into();
131
132 if time > JsDate::MAX_VALUE {
133 return Err(DateError(DateErrorKind::Overflow));
134 } else if time < JsDate::MIN_VALUE {
135 return Err(DateError(DateErrorKind::Underflow));
136 }
137
138 let local = unsafe { sys::date::new_date(env, time) };
139 let date = Handle::new_internal(JsDate(local));
140 Ok(date)
141 }
142
143 pub fn new_lossy<'a, C: Context<'a>, V: Into<f64>>(cx: &mut C, value: V) -> Handle<'a, JsDate> {
146 let env = cx.env().to_raw();
147 let local = unsafe { sys::date::new_date(env, value.into()) };
148 Handle::new_internal(JsDate(local))
149 }
150
151 pub fn value<'a, C: Context<'a>>(&self, cx: &mut C) -> f64 {
153 let env = cx.env().to_raw();
154 unsafe { sys::date::value(env, self.to_local()) }
155 }
156
157 pub fn is_valid<'a, C: Context<'a>>(&self, cx: &mut C) -> bool {
160 let value = self.value(cx);
161 (JsDate::MIN_VALUE..=JsDate::MAX_VALUE).contains(&value)
162 }
163}
164
165impl ValueInternal for JsDate {
166 fn name() -> &'static str {
167 "object"
168 }
169
170 fn is_typeof<Other: Value>(cx: &mut Cx, other: &Other) -> bool {
171 unsafe { sys::tag::is_date(cx.env().to_raw(), other.to_local()) }
172 }
173
174 fn to_local(&self) -> raw::Local {
175 self.0
176 }
177
178 unsafe fn from_local(_env: Env, h: raw::Local) -> Self {
179 JsDate(h)
180 }
181}
182
183impl Object for JsDate {}