UNPKG

@nlabs/gothamjs

Version:
207 lines (206 loc) 23.7 kB
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime"; /** * Copyright (c) 2025-Present, Nitrogen Labs, Inc. * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms. */ import { useEffect, useState } from 'react'; export const DatePicker = ({ initialDate = new Date().getTime(), maxDate, minDate, onDateSelect })=>{ const defaultDate = new Date(initialDate); const [selectedDate, setSelectedDate] = useState(defaultDate); const [currentMonth, setCurrentMonth] = useState(defaultDate.getMonth()); const [currentYear, setCurrentYear] = useState(defaultDate.getFullYear()); useEffect(()=>{ const updatedDate = new Date(initialDate); setSelectedDate(updatedDate); setCurrentMonth(updatedDate.getMonth()); setCurrentYear(updatedDate.getFullYear()); }, [ initialDate ]); const handlePrevMonth = ()=>{ if (currentMonth === 0) { setCurrentMonth(11); setCurrentYear(currentYear - 1); } else { setCurrentMonth(currentMonth - 1); } }; const handleNextMonth = ()=>{ if (currentMonth === 11) { setCurrentMonth(0); setCurrentYear(currentYear + 1); } else { setCurrentMonth(currentMonth + 1); } }; const handleMonthChange = (e)=>{ setCurrentMonth(parseInt(e.target.value, 10)); }; const handleYearChange = (e)=>{ setCurrentYear(parseInt(e.target.value, 10)); }; const isDateInRange = (date)=>{ const dateTimestamp = date.getTime(); if (minDate && dateTimestamp < minDate) { return false; } if (maxDate && dateTimestamp > maxDate) { return false; } return true; }; const handleDateSelect = (day)=>{ const newDate = new Date(currentYear, currentMonth, day); if (!isDateInRange(newDate)) { return; } setSelectedDate(newDate); if (onDateSelect) { onDateSelect(newDate.getTime()); } }; // Get month name const monthNames = [ 'January', 'February', 'March', 'April', 'May', 'June', 'July', 'August', 'September', 'October', 'November', 'December' ]; const getDaysInMonth = (year, month)=>{ return new Date(year, month + 1, 0).getDate(); }; const getFirstDayOfMonth = (year, month)=>{ return new Date(year, month, 1).getDay(); }; // Generate years for dropdown (range based on min/max dates) const generateYearOptions = ()=>{ let startYear = currentYear - 5; let endYear = currentYear + 5; if (minDate) { const minYear = new Date(minDate).getFullYear(); startYear = minYear; } if (maxDate) { const maxYear = new Date(maxDate).getFullYear(); endYear = maxYear; } return Array.from({ length: endYear - startYear + 1 }, (_, i)=>startYear + i); }; // Generate days for the calendar const daysInMonth = getDaysInMonth(currentYear, currentMonth); const firstDayOfMonth = getFirstDayOfMonth(currentYear, currentMonth); const days = []; // Add empty cells for days before the first day of the month for(let i = 0; i < firstDayOfMonth; i++){ days.push(/*#__PURE__*/ _jsx("div", { className: "h-8 w-8" }, `empty-${i}`)); } // Add days of the month for(let day = 1; day <= daysInMonth; day++){ const date = new Date(currentYear, currentMonth, day); const isSelected = selectedDate.getDate() === day && selectedDate.getMonth() === currentMonth && selectedDate.getFullYear() === currentYear; const isDisabled = !isDateInRange(date); days.push(/*#__PURE__*/ _jsx("button", { onClick: ()=>handleDateSelect(day), disabled: isDisabled, className: `h-7 w-7 text-sm rounded-full flex items-center justify-center ${isSelected ? 'bg-blue-500 text-white' : isDisabled ? 'text-gray-300 cursor-not-allowed' : 'hover:bg-gray-200'}`, children: day }, day)); } const yearOptions = generateYearOptions(); return /*#__PURE__*/ _jsxs("div", { className: "bg-white border border-gray-200 rounded-lg shadow-lg p-3", children: [ /*#__PURE__*/ _jsxs("div", { className: "flex justify-between items-center mb-2", children: [ /*#__PURE__*/ _jsx("button", { onClick: handlePrevMonth, className: "p-1 rounded-full hover:bg-gray-200 text-sm", children: "<" }), /*#__PURE__*/ _jsxs("div", { className: "flex space-x-1", children: [ /*#__PURE__*/ _jsx("select", { value: currentMonth, onChange: handleMonthChange, className: "border border-gray-300 rounded px-1.5 py-0.5 text-sm", children: monthNames.map((month, index)=>/*#__PURE__*/ _jsx("option", { value: index, children: month }, month)) }), /*#__PURE__*/ _jsxs("div", { className: "relative inline-flex items-center", children: [ /*#__PURE__*/ _jsx("select", { value: currentYear, onChange: handleYearChange, className: "px-1.5 py-0.5 text-sm appearance-none bg-transparent focus:outline-none pr-5 hover:text-blue-600 cursor-pointer", style: { MozAppearance: 'none', WebkitAppearance: 'none', scrollbarWidth: 'thin' }, children: yearOptions.map((year)=>/*#__PURE__*/ _jsx("option", { value: year, className: "text-sm", children: year }, year)) }), /*#__PURE__*/ _jsx("div", { className: "pointer-events-none absolute inset-y-0 right-0 flex items-center px-1 text-gray-700", children: /*#__PURE__*/ _jsx("svg", { className: "h-3 w-3 fill-current", xmlns: "http://www.w3.org/2000/svg", viewBox: "0 0 20 20", children: /*#__PURE__*/ _jsx("path", { d: "M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z" }) }) }) ] }) ] }), /*#__PURE__*/ _jsx("button", { onClick: handleNextMonth, className: "p-1 rounded-full hover:bg-gray-200 text-sm", children: ">" }) ] }), /*#__PURE__*/ _jsxs("div", { className: "grid grid-cols-7 gap-0.5", children: [ [ 'Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa' ].map((day)=>/*#__PURE__*/ _jsx("div", { className: "text-center font-medium text-gray-500 text-xs", children: day }, day)), days ] }) ] }); }; export default DatePicker; //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/nitrog7/Development/gothamjs/src/components/DateField/DatePicker.tsx"],"sourcesContent":["/**\n * Copyright (c) 2025-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {useEffect, useState} from 'react';\n\nimport type {FC, ReactNode} from 'react';\n\ninterface DatePickerProps {\n  initialDate?: number;\n  maxDate?: number;\n  minDate?: number;\n  onDateSelect?: (date: number) => void;\n}\n\nexport const DatePicker: FC<DatePickerProps> = ({\n  initialDate = (new Date()).getTime(),\n  maxDate,\n  minDate,\n  onDateSelect\n}) => {\n  const defaultDate = new Date(initialDate);\n  const [selectedDate, setSelectedDate] = useState<Date>(defaultDate);\n  const [currentMonth, setCurrentMonth] = useState<number>(defaultDate.getMonth());\n  const [currentYear, setCurrentYear] = useState<number>(defaultDate.getFullYear());\n\n  useEffect(() => {\n    const updatedDate = new Date(initialDate);\n    setSelectedDate(updatedDate);\n    setCurrentMonth(updatedDate.getMonth());\n    setCurrentYear(updatedDate.getFullYear());\n  }, [initialDate]);\n\n  const handlePrevMonth = () => {\n    if (currentMonth === 0) {\n      setCurrentMonth(11);\n      setCurrentYear(currentYear - 1);\n    } else {\n      setCurrentMonth(currentMonth - 1);\n    }\n  };\n\n  const handleNextMonth = () => {\n    if (currentMonth === 11) {\n      setCurrentMonth(0);\n      setCurrentYear(currentYear + 1);\n    } else {\n      setCurrentMonth(currentMonth + 1);\n    }\n  };\n\n  const handleMonthChange = (e: React.ChangeEvent<HTMLSelectElement>) => {\n    setCurrentMonth(parseInt(e.target.value, 10));\n  };\n\n  const handleYearChange = (e: React.ChangeEvent<HTMLSelectElement>) => {\n    setCurrentYear(parseInt(e.target.value, 10));\n  };\n\n  const isDateInRange = (date: Date): boolean => {\n    const dateTimestamp = date.getTime();\n\n    if (minDate && dateTimestamp < minDate) {\n      return false;\n    }\n\n    if (maxDate && dateTimestamp > maxDate) {\n      return false;\n    }\n\n    return true;\n  };\n\n  const handleDateSelect = (day: number) => {\n    const newDate = new Date(currentYear, currentMonth, day);\n\n    if (!isDateInRange(newDate)) {\n      return;\n    }\n\n    setSelectedDate(newDate);\n\n    if (onDateSelect) {\n      onDateSelect(newDate.getTime());\n    }\n  };\n\n  // Get month name\n  const monthNames = [\n    'January', 'February', 'March', 'April', 'May', 'June',\n    'July', 'August', 'September', 'October', 'November', 'December'\n  ];\n\n  const getDaysInMonth = (year: number, month: number) => {\n    return new Date(year, month + 1, 0).getDate();\n  };\n\n  const getFirstDayOfMonth = (year: number, month: number) => {\n    return new Date(year, month, 1).getDay();\n  };\n\n  // Generate years for dropdown (range based on min/max dates)\n  const generateYearOptions = () => {\n    let startYear = currentYear - 5;\n    let endYear = currentYear + 5;\n\n    if (minDate) {\n      const minYear = new Date(minDate).getFullYear();\n      startYear = minYear;\n    }\n\n    if (maxDate) {\n      const maxYear = new Date(maxDate).getFullYear();\n      endYear = maxYear;\n    }\n\n    return Array.from(\n      {length: endYear - startYear + 1},\n      (_, i) => startYear + i\n    );\n  };\n\n  // Generate days for the calendar\n  const daysInMonth = getDaysInMonth(currentYear, currentMonth);\n  const firstDayOfMonth = getFirstDayOfMonth(currentYear, currentMonth);\n\n  const days: ReactNode[] = [];\n  // Add empty cells for days before the first day of the month\n  for (let i = 0; i < firstDayOfMonth; i++) {\n    days.push(<div key={`empty-${i}`} className=\"h-8 w-8\"></div>);\n  }\n\n  // Add days of the month\n  for (let day = 1; day <= daysInMonth; day++) {\n    const date = new Date(currentYear, currentMonth, day);\n    const isSelected =\n      selectedDate.getDate() === day &&\n      selectedDate.getMonth() === currentMonth &&\n      selectedDate.getFullYear() === currentYear;\n\n    const isDisabled = !isDateInRange(date);\n\n    days.push(\n      <button\n        key={day}\n        onClick={() => handleDateSelect(day)}\n        disabled={isDisabled}\n        className={`h-7 w-7 text-sm rounded-full flex items-center justify-center ${\n          isSelected\n            ? 'bg-blue-500 text-white'\n            : isDisabled\n              ? 'text-gray-300 cursor-not-allowed'\n              : 'hover:bg-gray-200'\n        }`}\n      >\n        {day}\n      </button>\n    );\n  }\n\n  const yearOptions = generateYearOptions();\n\n  return (\n    <div className=\"bg-white border border-gray-200 rounded-lg shadow-lg p-3\">\n      <div className=\"flex justify-between items-center mb-2\">\n        <button\n          onClick={handlePrevMonth}\n          className=\"p-1 rounded-full hover:bg-gray-200 text-sm\"\n        >\n          &lt;\n        </button>\n        <div className=\"flex space-x-1\">\n          <select\n            value={currentMonth}\n            onChange={handleMonthChange}\n            className=\"border border-gray-300 rounded px-1.5 py-0.5 text-sm\"\n          >\n            {monthNames.map((month, index) => (\n              <option key={month} value={index}>\n                {month}\n              </option>\n            ))}\n          </select>\n          <div className=\"relative inline-flex items-center\">\n            <select\n              value={currentYear}\n              onChange={handleYearChange}\n              className=\"px-1.5 py-0.5 text-sm appearance-none bg-transparent focus:outline-none pr-5 hover:text-blue-600 cursor-pointer\"\n              style={{\n                MozAppearance: 'none',\n                WebkitAppearance: 'none',\n                scrollbarWidth: 'thin'\n              }}\n            >\n              {yearOptions.map(year => (\n                <option key={year} value={year} className=\"text-sm\">\n                  {year}\n                </option>\n              ))}\n            </select>\n            <div className=\"pointer-events-none absolute inset-y-0 right-0 flex items-center px-1 text-gray-700\">\n              <svg className=\"h-3 w-3 fill-current\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 20 20\">\n                <path d=\"M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z\" />\n              </svg>\n            </div>\n          </div>\n        </div>\n        <button\n          onClick={handleNextMonth}\n          className=\"p-1 rounded-full hover:bg-gray-200 text-sm\"\n        >\n          &gt;\n        </button>\n      </div>\n      <div className=\"grid grid-cols-7 gap-0.5\">\n        {['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'].map(day => (\n          <div key={day} className=\"text-center font-medium text-gray-500 text-xs\">\n            {day}\n          </div>\n        ))}\n        {days}\n      </div>\n    </div>\n  );\n};\n\nexport default DatePicker;"],"names":["useEffect","useState","DatePicker","initialDate","Date","getTime","maxDate","minDate","onDateSelect","defaultDate","selectedDate","setSelectedDate","currentMonth","setCurrentMonth","getMonth","currentYear","setCurrentYear","getFullYear","updatedDate","handlePrevMonth","handleNextMonth","handleMonthChange","e","parseInt","target","value","handleYearChange","isDateInRange","date","dateTimestamp","handleDateSelect","day","newDate","monthNames","getDaysInMonth","year","month","getDate","getFirstDayOfMonth","getDay","generateYearOptions","startYear","endYear","minYear","maxYear","Array","from","length","_","i","daysInMonth","firstDayOfMonth","days","push","div","className","isSelected","isDisabled","button","onClick","disabled","yearOptions","select","onChange","map","index","option","style","MozAppearance","WebkitAppearance","scrollbarWidth","svg","xmlns","viewBox","path","d"],"mappings":";AAAA;;;CAGC,GACD,SAAQA,SAAS,EAAEC,QAAQ,QAAO,QAAQ;AAW1C,OAAO,MAAMC,aAAkC,CAAC,EAC9CC,cAAc,AAAC,IAAIC,OAAQC,OAAO,EAAE,EACpCC,OAAO,EACPC,OAAO,EACPC,YAAY,EACb;IACC,MAAMC,cAAc,IAAIL,KAAKD;IAC7B,MAAM,CAACO,cAAcC,gBAAgB,GAAGV,SAAeQ;IACvD,MAAM,CAACG,cAAcC,gBAAgB,GAAGZ,SAAiBQ,YAAYK,QAAQ;IAC7E,MAAM,CAACC,aAAaC,eAAe,GAAGf,SAAiBQ,YAAYQ,WAAW;IAE9EjB,UAAU;QACR,MAAMkB,cAAc,IAAId,KAAKD;QAC7BQ,gBAAgBO;QAChBL,gBAAgBK,YAAYJ,QAAQ;QACpCE,eAAeE,YAAYD,WAAW;IACxC,GAAG;QAACd;KAAY;IAEhB,MAAMgB,kBAAkB;QACtB,IAAIP,iBAAiB,GAAG;YACtBC,gBAAgB;YAChBG,eAAeD,cAAc;QAC/B,OAAO;YACLF,gBAAgBD,eAAe;QACjC;IACF;IAEA,MAAMQ,kBAAkB;QACtB,IAAIR,iBAAiB,IAAI;YACvBC,gBAAgB;YAChBG,eAAeD,cAAc;QAC/B,OAAO;YACLF,gBAAgBD,eAAe;QACjC;IACF;IAEA,MAAMS,oBAAoB,CAACC;QACzBT,gBAAgBU,SAASD,EAAEE,MAAM,CAACC,KAAK,EAAE;IAC3C;IAEA,MAAMC,mBAAmB,CAACJ;QACxBN,eAAeO,SAASD,EAAEE,MAAM,CAACC,KAAK,EAAE;IAC1C;IAEA,MAAME,gBAAgB,CAACC;QACrB,MAAMC,gBAAgBD,KAAKvB,OAAO;QAElC,IAAIE,WAAWsB,gBAAgBtB,SAAS;YACtC,OAAO;QACT;QAEA,IAAID,WAAWuB,gBAAgBvB,SAAS;YACtC,OAAO;QACT;QAEA,OAAO;IACT;IAEA,MAAMwB,mBAAmB,CAACC;QACxB,MAAMC,UAAU,IAAI5B,KAAKW,aAAaH,cAAcmB;QAEpD,IAAI,CAACJ,cAAcK,UAAU;YAC3B;QACF;QAEArB,gBAAgBqB;QAEhB,IAAIxB,cAAc;YAChBA,aAAawB,QAAQ3B,OAAO;QAC9B;IACF;IAEA,iBAAiB;IACjB,MAAM4B,aAAa;QACjB;QAAW;QAAY;QAAS;QAAS;QAAO;QAChD;QAAQ;QAAU;QAAa;QAAW;QAAY;KACvD;IAED,MAAMC,iBAAiB,CAACC,MAAcC;QACpC,OAAO,IAAIhC,KAAK+B,MAAMC,QAAQ,GAAG,GAAGC,OAAO;IAC7C;IAEA,MAAMC,qBAAqB,CAACH,MAAcC;QACxC,OAAO,IAAIhC,KAAK+B,MAAMC,OAAO,GAAGG,MAAM;IACxC;IAEA,6DAA6D;IAC7D,MAAMC,sBAAsB;QAC1B,IAAIC,YAAY1B,cAAc;QAC9B,IAAI2B,UAAU3B,cAAc;QAE5B,IAAIR,SAAS;YACX,MAAMoC,UAAU,IAAIvC,KAAKG,SAASU,WAAW;YAC7CwB,YAAYE;QACd;QAEA,IAAIrC,SAAS;YACX,MAAMsC,UAAU,IAAIxC,KAAKE,SAASW,WAAW;YAC7CyB,UAAUE;QACZ;QAEA,OAAOC,MAAMC,IAAI,CACf;YAACC,QAAQL,UAAUD,YAAY;QAAC,GAChC,CAACO,GAAGC,IAAMR,YAAYQ;IAE1B;IAEA,iCAAiC;IACjC,MAAMC,cAAchB,eAAenB,aAAaH;IAChD,MAAMuC,kBAAkBb,mBAAmBvB,aAAaH;IAExD,MAAMwC,OAAoB,EAAE;IAC5B,6DAA6D;IAC7D,IAAK,IAAIH,IAAI,GAAGA,IAAIE,iBAAiBF,IAAK;QACxCG,KAAKC,IAAI,eAAC,KAACC;YAAuBC,WAAU;WAAxB,CAAC,MAAM,EAAEN,GAAG;IAClC;IAEA,wBAAwB;IACxB,IAAK,IAAIlB,MAAM,GAAGA,OAAOmB,aAAanB,MAAO;QAC3C,MAAMH,OAAO,IAAIxB,KAAKW,aAAaH,cAAcmB;QACjD,MAAMyB,aACJ9C,aAAa2B,OAAO,OAAON,OAC3BrB,aAAaI,QAAQ,OAAOF,gBAC5BF,aAAaO,WAAW,OAAOF;QAEjC,MAAM0C,aAAa,CAAC9B,cAAcC;QAElCwB,KAAKC,IAAI,eACP,KAACK;YAECC,SAAS,IAAM7B,iBAAiBC;YAChC6B,UAAUH;YACVF,WAAW,CAAC,8DAA8D,EACxEC,aACI,2BACAC,aACE,qCACA,qBACN;sBAED1B;WAXIA;IAcX;IAEA,MAAM8B,cAAcrB;IAEpB,qBACE,MAACc;QAAIC,WAAU;;0BACb,MAACD;gBAAIC,WAAU;;kCACb,KAACG;wBACCC,SAASxC;wBACToC,WAAU;kCACX;;kCAGD,MAACD;wBAAIC,WAAU;;0CACb,KAACO;gCACCrC,OAAOb;gCACPmD,UAAU1C;gCACVkC,WAAU;0CAETtB,WAAW+B,GAAG,CAAC,CAAC5B,OAAO6B,sBACtB,KAACC;wCAAmBzC,OAAOwC;kDACxB7B;uCADUA;;0CAKjB,MAACkB;gCAAIC,WAAU;;kDACb,KAACO;wCACCrC,OAAOV;wCACPgD,UAAUrC;wCACV6B,WAAU;wCACVY,OAAO;4CACLC,eAAe;4CACfC,kBAAkB;4CAClBC,gBAAgB;wCAClB;kDAECT,YAAYG,GAAG,CAAC7B,CAAAA,qBACf,KAAC+B;gDAAkBzC,OAAOU;gDAAMoB,WAAU;0DACvCpB;+CADUA;;kDAKjB,KAACmB;wCAAIC,WAAU;kDACb,cAAA,KAACgB;4CAAIhB,WAAU;4CAAuBiB,OAAM;4CAA6BC,SAAQ;sDAC/E,cAAA,KAACC;gDAAKC,GAAE;;;;;;;;kCAKhB,KAACjB;wBACCC,SAASvC;wBACTmC,WAAU;kCACX;;;;0BAIH,MAACD;gBAAIC,WAAU;;oBACZ;wBAAC;wBAAM;wBAAM;wBAAM;wBAAM;wBAAM;wBAAM;qBAAK,CAACS,GAAG,CAACjC,CAAAA,oBAC9C,KAACuB;4BAAcC,WAAU;sCACtBxB;2BADOA;oBAIXqB;;;;;AAIT,EAAE;AAEF,eAAelD,WAAW"}