Javascript.RU

Создать новую тему Ответ
 
Опции темы Искать в теме
  #1 (permalink)  
Старый 06.06.2021, 01:01
Кандидат Javascript-наук
Отправить личное сообщение для Besprizornik Посмотреть профиль Найти все сообщения от Besprizornik
 
Регистрация: 24.02.2019
Сообщений: 126

как ограничить опции в селекте разному числу дней в месяце?
скрипт выводит в селектах 31 опцию для дней недели, как сделать чтобы для апреля июня сентября ноября было по 30 опций, а в феврале 28 или 29 в зависимости от года, высокосный или нет, и учесть что нулевого года нету, 1ый 5ый 9ый и т.д. год до н.э. высокосный, а также учесть что в старом стиле все года кратные 4 высокосные, нет условия что года кратные 100 простые и только красные 400 высокосные

<!DOCTYPE html>
<html>
<head>
    <meta charset="UTF-8">
    <title>Конвертирование григорианского/юлианского календаря</title>
    <style>
    @import url('https://fonts.googleapis.com/css2?family=Old+Standard+TT:wght@700&family=Tinos:ital@1&display=swap');
    a { text-decoration: none; color: hsl(210,100%,50%); }
    form { margin: 0; }
    div { border: 1px solid hsl(240,100%,50%); color: hsl(240,100%,50%); font: 16px 'Old Standard TT', serif; padding: 5px 10px; overflow: auto; min-width: 504px; margin: 0 0 5px 0; }
    span { font-family: 'Tinos', serif; }
    h1 { margin: 0; color: hsl(0,0%,100%); font: 16px 'Old Standard TT', serif; text-shadow: 1px 1px hsl(240,100%,50%), -1px 1px hsl(240,100%,50%), 1px -1px hsl(240,100%,50%), -1px -1px hsl(240,100%,50%), 1px 0px hsl(240,100%,50%), 0px 1px hsl(240,100%,50%), -1px 0px hsl(240,100%,50%), 0px -1px hsl(240,100%,50%); -webkit-animation: h-pulsare 1.35s linear infinite; animation: h-pulsare 1.35s linear infinite; }
    button { font: 16px 'Old Standard TT', serif; background: hsl(240,100%,97%); color: hsl(240,100%,50%); border: 1px solid hsl(240,100%,50%); padding: 5px 25px; }
    #b { margin: 0; }
    #gDay {  }
    #gMonth {  }
    #gYear { width: 56px; display: inline-block; text-align: center; }
    #gEra {  }
    #jwDay {  }
    #jDay {  }
    #jMonth {  }
    #jYear { width: 56px; display: inline-block; text-align: center; }
    #jEra {  }
    #gwDay option, #gDay, #gMonth, #gYear, #gEra, #jwDay, #jDay, #jMonth, #jYear, #jEra { background-color: hsl(240, 100%, 93%); color: hsl(240,100%,50%); }
    select, input#gYear, input#jYear, input { background: hsl(240,100%,97%); color: hsl(240,100%,50%); border: 1px solid hsl(240,100%,50%); font: 16px 'Old Standard TT', serif; line-height: 1; margin: 5px 3px; }
    input { background: hsl(240,100%,97%); color: hsl(240,100%,50%); border: 1px solid hsl(240,100%,50%); font: 16px 'Old Standard TT', serif; line-height: 1.6; margin: 5px 3px; }
    @-webkit-keyframes h-pulsare { 0%, 100% { color: hsl(0,0%,100%); text-shadow: 1px 1px hsl(240,100%,50%), -1px 1px hsl(240,100%,50%), 1px -1px hsl(240,100%,50%), -1px -1px hsl(240,100%,50%), 1px 0px hsl(240,100%,50%), 0px 1px hsl(240,100%,50%), -1px 0px hsl(240,100%,50%), 0px -1px hsl(240,100%,50%); } 50% { color: hsl(240,100%,90%); text-shadow: 1px 1px hsl(210,100%,50%), -1px 1px hsl(210,100%,50%), 1px -1px hsl(210,100%,50%), -1px -1px hsl(210,100%,50%), 1px 0px hsl(210,100%,50%), 0px 1px hsl(210,100%,50%), -1px 0px hsl(210,100%,50%), 0px -1px hsl(210,100%,50%); } } @keyframes h-pulsare { 0%, 100% { color: hsl(0,0%,100%); text-shadow: 1px 1px hsl(240,100%,50%), -1px 1px hsl(240,100%,50%), 1px -1px hsl(240,100%,50%), -1px -1px hsl(240,100%,50%), 1px 0px hsl(240,100%,50%), 0px 1px hsl(240,100%,50%), -1px 0px hsl(240,100%,50%), 0px -1px hsl(240,100%,50%); } 50% { color: hsl(240,100%,90%); text-shadow: 1px 1px hsl(210,100%,50%), -1px 1px hsl(210,100%,50%), 1px -1px hsl(210,100%,50%), -1px -1px hsl(210,100%,50%), 1px 0px hsl(210,100%,50%), 0px 1px hsl(210,100%,50%), -1px 0px hsl(210,100%,50%), 0px -1px hsl(210,100%,50%); } }
    </style>
</head>
<body>
<div id="h">
<h1>Конвертирование дат из григорианского календаря в юлианский и обратно.</h1>
</div>
<form name="converter">
    <div id="g">
    <label>по григорианскому календарю <span>(новый стиль)</span></label><br/>
    <select name="gwDay"></select><br>
    <select name="gDay"></select>
    <select name="gMonth"></select>
    <input type="number" name="gYear" id="gYear" min="-4713">
    <select name="gEra"></select><br/>
    <span>не ранее 24 ноября 4714 г. до н. э.</span><br/>
    <input type="button" name="gCalc" value="конвертировать в юлианский календарь">
    </div>
    <div id="u">
    <label>по юлианскому календарю <span>(старый стиль)</span></label><br/>
    <select name="jwDay"></select><br>
    <select name="jDay"></select>
    <select name="jMonth"></select>
    <input type="number" name="jYear" id="jYear" min="-4712">
    <select name="jEra"></select><br/>
    <span>не ранее 1 января 4713 г. до н.э.</span><br/>
    <input type="button" name="jCalc" value="конвертировать в григорианский календарь">
    </div>
    <div id="s">
    <button type="button" class="express" value="g_n_o_w_ce">сбросить на текущую дату</button>
    </div>
    <div id="b">
    <small>
    * Алгоритм действителен для всех (возможно, пролептических) григорианских дат не ранее 24 ноября 4714 до н. э. и юлианских дат не ранее 1 января 4713 до н. э.<br/></small>
    <a href="https://en.wikipedia.org/wiki/Julian_day">https://en.wikipedia.org/wiki/Julian_day</a>
    </div>
</form>
<script>
document.addEventListener('DOMContentLoaded', function () {
// https://en.wikipedia.org/wiki/Julian_day#Julian_day_number_calculation
// The algorithms are valid for JDN >= 0, Gregorian >= 24.11.-4713, Julian >= 01.01.-4712
function gregorianToJdn(d, m, y) {
    return tr((1461 * (y + 4800 + tr((m - 14) / 12))) / 4) +
           tr((367 * (m - 2 - 12 * tr((m - 14) / 12))) / 12) -
           tr((3 * tr((y + 4900 + tr((m - 14) / 12)) / 100)) / 4) + d - 32075;
}
function julianToJdn(d, m, y) {
    return 367 * y - tr((7 * (y + 5001 + tr((m - 9) / 7))) / 4) + tr((275 * m) / 9) + d + 1729777;
}
function jdnToJulianOrGregorian(j, to) { // to: true (julian), false (gregorian)
    var f = (to) ? (j + 1401) : (j + 1363 + tr((tr((4 * j + 274277) / 146097) * 3) / 4));
    var e = 4 * f + 3, h = 5 * tr((e % 1461) / 4) + 2;
    var d = tr((h % 153) / 5) + 1, m = ((tr(h / 153) + 2) % 12) + 1, y = tr(e / 1461) - 4716 + tr((14 - m) / 12);
    return { d: d, m: m, y: y };
}
function jdnToWeekDay(j) { return (j % 7) + 1; }
function tr(n) { return n - n % 1; } // truncate (for integer division)
// -- wiki end --
var C = document.converter;
// options
var wdays = ['Понедельник','Вторник','Среда','Четверг','Пятница','Суббота','Воскресенье'];
for (var o = '', i = 0; i < wdays.length; i++) o += '<option value="' + (i + 1) + '">' + wdays[i] + '</option>';
C.gwDay.innerHTML = C.jwDay.innerHTML = o;
for (var o = '', i = 1; i <= 31; i++) o += '<option value="' + i  + '">' + i + '</option>';
C.gDay.innerHTML = C.jDay.innerHTML = o;
var months = ['января','февраля','марта','апреля','мая','июня','июля','августа','сентября','октября','ноября','декабря'];
for (var o = '', i = 0; i < months.length; i++) o += '<option value="' + (i + 1) + '">' + months[i] + '</option>';
C.gMonth.innerHTML = C.jMonth.innerHTML = o;
var eras = ['н. э.','до н. э.'], _eras = ['ce','bce'];
for (var o = '', i = 0; i < eras.length; i++) o += '<option value="' + _eras[i] + '">' + eras[i] + '</option>';
C.gEra.innerHTML = C.jEra.innerHTML = o;
// -- options end --
C.gCalc.onclick = function () {
    var gd = +C.gDay.value, gm = +C.gMonth.value, gy = historicalToAstronomical(+C.gYear.value, C.gEra.value);
    var jdn = gregorianToJdn(gd, gm, gy), wd = jdnToWeekDay(jdn), jd = jdnToJulianOrGregorian(jdn, true);
    var jy = astronomicalToHistorical(jd.y);
    C.gwDay.value = C.jwDay.value = wd, C.jDay.value = jd.d, C.jMonth.value = jd.m, C.jYear.value = jy.y, C.jEra.value = jy.e;
    onUpdateAll(jdn);
};
C.jCalc.onclick = function () {
    var jd = +C.jDay.value, jm = +C.jMonth.value, jy = historicalToAstronomical(+C.jYear.value, C.jEra.value);
    var jdn = julianToJdn(jd, jm, jy), wd = jdnToWeekDay(jdn), gd = jdnToJulianOrGregorian(jdn, false);
    var gy = astronomicalToHistorical(gd.y);
    C.gwDay.value = C.jwDay.value = wd, C.gDay.value = gd.d, C.gMonth.value = gd.m, C.gYear.value = gy.y, C.gEra.value = gy.e;
    onUpdateAll(jdn);
};
function initDate(c, d, m, y, e) {
    if (d + m + y == 'now') { var n = new Date(); d = n.getDate(), m = n.getMonth() + 1, y = n.getFullYear(); }
    C[c + 'Day'].value = d, C[c + 'Month'].value = m, C[c + 'Year'].value = y, C[c + 'Era'].value = e;
    C[c + 'Calc'].onclick();
    }
    function historicalToAstronomical(y, e) { // e: 'ce', 'bce'
    e = (y < 1) ? 'bce' : e;
    return ((e == 'bce') && y > 0) ? (1 - y) : y;
    }
    function astronomicalToHistorical(y) {
    var e = (y < 1) ? 'bce' : 'ce';
    return { y: (e == 'bce') ? -(y - 1) : y, e: e };
    }
    for (var ess = C.querySelectorAll('.express'), es, i = 0; i < ess.length; i++) {
    es = ess[i], es.onclick = Function.prototype.apply.bind(initDate, null, es.value.split('_'));
    }
    // Express-url format ...?C=D_M_Y_E
    var m = location.search.match(/(j|g)=(\d+)_(\d+)_(\d+)_(ce|bce)/);
    if (m) { initDate(m[1], m[2], m[3], m[4], m[5]); } else { initDate('g', 'n', 'o', 'w', 'ce'); }
    function onUpdateAll(jdn) {
    console.log('JDN:', jdn);
    console.assert(jdn >= 2299161, 'Proleptic: Gn < 15.10.1582 CE');
    console.assert(jdn >= 1704987, 'Proleptic: Jn < 01.01.45 BCE');
    console.assert(jdn >= 0, 'Invalid: Gn < 24.11.4714 BCE, Jn < 01.01.4713 BCE');
    }
    });
    </script>
    </body>
    </html>
Ответить с цитированием
  #2 (permalink)  
Старый 06.06.2021, 17:46
Аватар для ksa
ksa ksa вне форума
CacheVar
Отправить личное сообщение для ksa Посмотреть профиль Найти все сообщения от ksa
 
Регистрация: 19.08.2010
Сообщений: 14,228

Сообщение от Besprizornik
скрипт выводит в селектах 31 опцию для дней недели, как сделать чтобы для апреля июня сентября ноября было по 30 опций, а в феврале 28 или 29 в зависимости от года, высокосный или нет
Если хочешь узнать количество дней в некоем месяце, некоего года - возьми первое число следующего за ним месяца и отними 1 день.
Потом узнаешь день этой даты... Это твое искомое число.
Ответить с цитированием
  #3 (permalink)  
Старый 06.06.2021, 17:49
Аватар для ksa
ksa ksa вне форума
CacheVar
Отправить личное сообщение для ksa Посмотреть профиль Найти все сообщения от ksa
 
Регистрация: 19.08.2010
Сообщений: 14,228

Даже в учебнике есть пример как найти "последний день месяца"...
https://learn.javascript.ru/task/last-day-of-month
Ответить с цитированием
  #4 (permalink)  
Старый 06.06.2021, 19:15
Аватар для рони
Профессор
Отправить личное сообщение для рони Посмотреть профиль Найти все сообщения от рони
 
Регистрация: 27.05.2010
Сообщений: 33,126

Besprizornik,
https://javascript.ru/forum/misc/476...v-mesyace.html
Ответить с цитированием
  #5 (permalink)  
Старый 09.06.2021, 20:48
Кандидат Javascript-наук
Отправить личное сообщение для Besprizornik Посмотреть профиль Найти все сообщения от Besprizornik
 
Регистрация: 24.02.2019
Сообщений: 126

ksa,
рони,
эти варианты покажут неверный ответ для февраля 1900, 2100 г. для годов делящихся на 100 и не делящих на 400 без остатка, для старого стиля.
Ответить с цитированием
  #6 (permalink)  
Старый 09.06.2021, 20:49
Аватар для ksa
ksa ksa вне форума
CacheVar
Отправить личное сообщение для ksa Посмотреть профиль Найти все сообщения от ksa
 
Регистрация: 19.08.2010
Сообщений: 14,228

Besprizornik, где твой тестовый пример?
Ответить с цитированием
Ответ



Опции темы Искать в теме
Искать в теме:

Расширенный поиск


Похожие темы
Тема Автор Раздел Ответов Последнее сообщение
Таймер с учетом дней в месяце. SinteC Элементы интерфейса 2 05.12.2015 08:39
Как вывести колличество дней в месяце? akkord87 Общие вопросы Javascript 2 04.06.2014 01:34
Как вы относитесь к наркоманам? Maxmaxmaximus7 Оффтопик 7 05.02.2014 13:29
Управление скроллом "а-ля тач" HonesT Элементы интерфейса 2 27.08.2013 14:25
Как получить количество дней до определенной даты beard Общие вопросы Javascript 3 24.06.2012 17:22