ID:               33445
 Updated by:       [EMAIL PROTECTED]
 Reported By:      richard dot quadling at bandvulc dot co dot uk
-Status:           Assigned
+Status:           Closed
 Bug Type:         Feature/Change Request
 Operating System: All
 PHP Version:      5.0.4
 Assigned To:      derick
 New Comment:

Already implemented in PHP 5.1 as date('o').


Previous Comments:
------------------------------------------------------------------------

[2005-06-23 13:32:10] [EMAIL PROTECTED]

This will be re-implemented very soon, so stay tuned!

------------------------------------------------------------------------

[2005-06-23 10:44:16] richard dot quadling at bandvulc dot co dot uk

Maybe a better solution would be to make the strftime and date
functions the same and to consolidate the code to remove the dependency
on platform specific libraries (e.g. the Windows strftime function does
not support V,G or g).

Alternatively, amend the documentation to indicate which formats are
supported by which platform.

It does seem that both of these functions do the same job and could be
consilidated though.

Richard.

------------------------------------------------------------------------

[2005-06-23 10:17:55] richard dot quadling at bandvulc dot co dot uk

Description:
------------
This could also be a fix for a "Will Not Fix" bug #22711.



On Windows, the strftime() function does not support %V, %G or %g.

The date() function supports 'W' which deals with %V (if you get what I
mean), but when the week number is 52 or 53 and you are looking at a
January date, getting the correct year also would be useful.

I've included below a CVS diff datetime.c.

I don't know who to send it to.

date('V') emulates strftime('%G');
date('v') emulates strftime('%g');

I've also rewritten the date('W') code to combine with V and v.

Index: datetime.c
===================================================================
RCS file: /repository/php-src/ext/standard/datetime.c,v
retrieving revision 1.129
diff -u -r1.129 datetime.c
--- datetime.c  19 Jun 2005 22:15:26 -0000      1.129
+++ datetime.c  23 Jun 2005 08:00:48 -0000
@@ -288,7 +288,7 @@
        pval **format, **timestamp;
        time_t the_time;
        struct tm *ta, tmbuf;
-       int i, size = 0, length, h, beat, fd, wd, yd, wk;
+       int i, size = 0, length, h, beat, fd, wd, yd, wk, yr;
        char tmp_buff[32];
 #if !HAVE_TM_GMTOFF
        long tzone;
@@ -382,6 +382,7 @@
                                size += 5;
                                break;
                        case 'Y':               /* year, numeric, 4 digits */
+                       case 'V':               /* ISO-8601 year number of 
year, numeric, 4 digits */
                                size += 4;
                                break;
                        case 'M':               /* month, textual, 3 letters */
@@ -406,6 +407,7 @@
                        case 'S':               /* standard english suffix for 
the day of the month
(e.g. 3rd, 2nd, etc) */
                        case 't':               /* days in current month */
                        case 'W':               /* ISO-8601 week number of 
year, weeks starting on
Monday */
+                       case 'v':               /* ISO-8601 year number of 
year, numeric, 2 digits */
                                size += 2;
                                break;
                        case '\\':
@@ -641,27 +643,67 @@
                                strcat(Z_STRVAL_P(return_value), tmp_buff);
                                break;
                        case 'W':               /* ISO-8601 week number of 
year, weeks starting on
Monday */
-                               wd = ta->tm_wday == 0 ? 6 : ta->tm_wday - 1; /* 
weekday */
-                               yd = ta->tm_yday + 1;                           
        /* days since January 1st */
-
-                               fd = (7 + wd - yd % 7+ 1) % 7;                  
/* weekday (1st January) */     
-
-                               /* week is a last year week (52 or 53) */
-                               if ((yd <= 7 - fd) && fd > 3){                  
-                                       wk = (fd == 4 || (fd == 5 && 
isleap((ta->tm_year + YEAR_BASE -
1)))) ? 53 : 52;
-                               }
-                               /* week is a next year week (1) */
-                               else if (isleap((ta->tm_year+YEAR_BASE)) + 365 
- yd < 3 - wd){
-                                       wk = 1;
+                       case 'V':               /* ISO-8601 year number of 
year, numeric, 4 digits */
+                       case 'v':               /* ISO-8601 year number of 
year, numeric, 2 digits */
+                               yr = ta->tm_year + YEAR_BASE;
+                               yd = ta->tm_yday;
+                               wd = ta->tm_wday;
+                               while(1) {
+                                       int len, bot, top;
+                                       
+                                       len = isleap(yr) ? 366 : 365;
+                                       bot = ((yd + 11 - wd) % 7) - 3;
+                                       top = bot - (len % 7);
+                                       if (top < -3) {
+                                               top += 7;
+                                       }
+                                       top += len;
+                                       if (yd >= top) {
+                                               ++yr;
+                                               w = 1;
+                                               break;
+                                       }
+                                       if (yd >= bot) {
+                                               w = 1 + ((yd - bot) / 7);
+                                               break;
+                                       }
+                                       --year;
+                                       yd += isleap(yr) ? 366 : 365;
                                }
-                               /* normal week */
-                               else {
-                                       wk = (yd + 6 - wd + fd) / 7 - (fd > 3);
+                               switch (Z_STRVAL_PP(format)[i]) {
+                                       case 'W':               /* ISO-8601 
week number of year, weeks starting on
Monday */
+                                               sprintf(tmp_buff, "%d", wk);  
/* SAFE */
+                                               break;
+                                       case 'V':               /* ISO-8601 
year number of year, numeric, 4 digits */
+                                               sprintf(tmp_buff, "%d", yr);  
/* SAFE */
+                                               break;
+                                       case 'v':               /* ISO-8601 
year number of year, numeric, 2 digits */
+                                               sprintf(tmp_buff, "%02d", yr % 
100);  /* SAFE */
+                                               break;
                                }
-
-                               sprintf(tmp_buff, "%d", wk);  /* SAFE */
                                strcat(Z_STRVAL_P(return_value), tmp_buff);
                                break;
+//                             wd = ta->tm_wday == 0 ? 6 : ta->tm_wday - 1; /* 
weekday */
+//                             yd = ta->tm_yday + 1;                           
        /* days since January 1st */
+
+//                             fd = (7 + wd - yd % 7+ 1) % 7;                  
/* weekday (1st January) */     
+
+//                             /* week is a last year week (52 or 53) */
+//                             if ((yd <= 7 - fd) && fd > 3){                  
+//                                     wk = (fd == 4 || (fd == 5 && 
isleap((ta->tm_year + YEAR_BASE -
1)))) ? 53 : 52;
+//                             }
+//                             /* week is a next year week (1) */
+//                             else if (isleap((ta->tm_year+YEAR_BASE)) + 365 
- yd < 3 - wd){
+//                                     wk = 1;
+//                             }
+//                             /* normal week */
+//                             else {
+//                                     wk = (yd + 6 - wd + fd) / 7 - (fd > 3);
+//                             }
+
+//                             sprintf(tmp_buff, "%d", wk);  /* SAFE */
+//                             strcat(Z_STRVAL_P(return_value), tmp_buff);
+//                             break;
 
                        default:
                                length = strlen(Z_STRVAL_P(return_value));
@@ -773,22 +815,62 @@
                case 'I':
                        return ta->tm_isdst;
                case 'W':               /* ISO-8601 week number of year, weeks 
starting on Monday
*/
-                       wd = (ta->tm_wday == 0) ? 6 : ta->tm_wday - 1; /* 
weekday */
-                       yd = ta->tm_yday + 1;                                   
/* days since January 1st */
-                       fd = (7 + wd - yd % 7+ 1) % 7;                  /* 
weekday (1st January) */
-                       if ((yd <= 7 - fd) && fd > 3) {                 /* week 
is a last year week (52
or 53) */
-                               wk = (fd == 4 || (fd == 5 && 
isleap((ta->tm_year + YEAR_BASE -
1)))) ? 53 : 52;
-                       }
-                       /* week is a next year week (1) */
-                       else if (isleap((ta->tm_year + YEAR_BASE)) + 365 - yd < 
3 - wd) {
-                               wk = 1;
+               case 'V':               /* ISO-8601 year number of year, 
numeric, 4 digits */
+               case 'v':               /* ISO-8601 year number of year, 
numeric, 2 digits */
+                       yr = ta->tm_year + YEAR_BASE;
+                       yd = ta->tm_yday;
+                       wd = ta->tm_wday;
+                       while(1) {
+                               int len, bot, top;
+                               
+                               len = isleap(yr) ? 366 : 365;
+                               bot = ((yd + 11 - wd) % 7) - 3;
+                               top = bot - (len % 7);
+                               if (top < -3) {
+                                       top += 7;
+                               }
+                               top += len;
+                               if (yd >= top) {
+                                       ++yr;
+                                       w = 1;
+                                       break;
+                               }
+                               if (yd >= bot) {
+                                       w = 1 + ((yd - bot) / 7);
+                                       break;
+                               }
+                               --year;
+                               yd += isleap(yr) ? 366 : 365;
                        }
-                       /* normal week */
-                       else {
-                               wk = (yd + 6 - wd + fd) / 7 - (fd > 3);
+                       switch (format) {
+                               case 'W':               /* ISO-8601 week number 
of year, weeks starting on
Monday */
+                                       return wk;
+                                       break;
+                               case 'V':               /* ISO-8601 year number 
of year, numeric, 4 digits */
+                                       return yr;
+                                       break;
+                               case 'v':               /* ISO-8601 year number 
of year, numeric, 2 digits */
+                                       yr = yr % 100
+                                       return yr;
+                                       break;
                        }
-                       return wk;
                        break;
+//                     wd = (ta->tm_wday == 0) ? 6 : ta->tm_wday - 1; /* 
weekday */
+//                     yd = ta->tm_yday + 1;                                   
/* days since January 1st */
+//                     fd = (7 + wd - yd % 7+ 1) % 7;                  /* 
weekday (1st January) */
+//                     if ((yd <= 7 - fd) && fd > 3) {                 /* week 
is a last year week (52
or 53) */
+//                             wk = (fd == 4 || (fd == 5 && 
isleap((ta->tm_year + YEAR_BASE -
1)))) ? 53 : 52;
+//                     }
+//                     /* week is a next year week (1) */
+//                     else if (isleap((ta->tm_year + YEAR_BASE)) + 365 - yd < 
3 - wd)
{
+//                             wk = 1;
+//                     }
+//                     /* normal week */
+//                     else {
+//                             wk = (yd + 6 - wd + fd) / 7 - (fd > 3);
+//                     }
+//                     return wk;
+//                     break;
                default:
                        return 0;
        }


Regards,

Richard Quadling.




------------------------------------------------------------------------


-- 
Edit this bug report at http://bugs.php.net/?id=33445&edit=1

Reply via email to