Wrong date on setting Calendar timeZone to UTC - ios

Allover the app I use Date objects that when I NSLog the value it shows me:
2020-05-24 22:00:00 +0000
Which I think locally means the 25th (- 1 for summer, -1 for timezone). I want to do some Calendar date comparisons:
var calendar = Calendar(identifier: .gregorian)
calendar.timeZone = TimeZone(identifier: "UTC")!
// In order to have start on Monday
calendar.firstWeekday = 2
Using this calendar, lets say I want to get the starting date of current week:
extension Date
{
var startOfWeek: Date {
return Calendar.gregorian.date(from: Calendar.gregorian.dateComponents([.yearForWeekOfYear, .weekOfYear], from: self))!
}
}
If I NSLog:
Date().startOfWeek
It will show me:
2020-05-25 00:00:00 +0000
If I disable the timeZone line on Calendar, it shows me:
2020-05-24 22:00:00 +0000
I always thought the second one is the correct UTC version. Am I wrong? Because I thought all core data dates, all dates are in the 2nd version. In short: If I set Calendar to UTC, my date comparissons are wrong. If I don't they are good. And all this time dates are in UTC.

You are wrong because CoreData dates are not affected by TimeZone. Dates are dates. Think of them as numeric values. When you translate that value to a date and hour then, and only then, the TimeZone is applied.
In your example everything is correct. For a calendar whose TimeZone is UTC, 2020-05-25 00:00:00 +0000 is the beginning of the week. If you use other TimeZone values (for example the default value from Locale) then the your week start at 2020-05-24 22:00:00 +0000. That means that in your TimeZone the hour is 2020-05-25 00:00:00.

Related

Adding time from string into date object swift

I have a time coming from service in string form like so "12:30 PM".
I am able to get this time but I want to add this time in current date. like so
31/3/2021 12:30 PM
The current date is in date object and coming time is in string format.
Please let me know what is a right way to do so? I am right now taking 12, 30,from string and setting it via Calendar. But dont know how to set am pm . Please let me know how to append time with date object. Thanks in advance.
First, if that is the date string you are getting from your service, it is incomplete. It needs a time zone.
Here's what I would do:
Assuming the service always uses the same time zone, find out that time zone.
Create a date formatter for that date string format, including the AM/PM bit.
Set the date formatter to use the time zone from step 1.
Convert the date string to a Date object using your DateFormatter.
Use the current calendar to extract the hours and minutes values into a DateComponents object.
Get the current date, and use the Calendar function date(bySettingHour:minute:second:of:matchingPolicy:repeatedTimePolicy:direction:) to set the hour and minutes of the current date to the values you got from step 5.
You should search in the Xcode help system for:
Dates and Times (overview)
Calendrical calculations (discussion specific to doing math on dates and times)
DateFormatter. (See this article for info on the characters to use to build your dateFormat string.)
Calendar
DateComponents
Calendar
You can do this by setting up your DateFormatter with the correct timeZone, calendar, defaultDate, and dateFormat. Here's an example:
import Foundation
let parser = DateFormatter()
parser.calendar = Calendar(identifier: .gregorian)
parser.timeZone = TimeZone(identifier: "US/Eastern")!
parser.defaultDate = parser.calendar!.date(
from: DateComponents(
timeZone: parser.timeZone!, era: 1,
year: 2021, month: 3, day: 31,
hour: 12, minute: 0, second: 0))!
parser.dateFormat = "hh:mm a"
print(parser.date(from: "12:30 PM"))
Output:
Optional(2021-03-31 16:30:00 +0000)

Check if date has same day in swift

Hi there in my app I need to filter some documents through date. I need to see if the first date is the same day of the second date, so I searched on Apple Documentation to see if there are a solution to do this without create a method and I found this instruction isDate(_:inSameDayAs:), but if i try to compare the following date:
2020-07-29 16:15:50 +0000
2020-07-29 22:00:00 +0000
As you can see the day is the same, but I'm not able to understand why it return false, what's wrong?
CODE
Here's my code to check the difference between days:
myArray.filter({Calendar.current.isDate($0.log.createdDate, inSameDayAs:date)})
Date represents instants in time. Two instants of time could be in the same day in one timezone, but not in the same day in another timezone. These two instants in time:
2020-07-29 16:15:50 +0000
2020-07-29 22:00:00 +0000
are in the same day in the UTC timezone. However, in a timezone where the offset is 5 hours ahead of UTC (UTC+5) for example, the two times will not be in the same day, because they will become:
2020-07-29 21:15:50
2020-07-30 03:00:00
in that timezone
Now you should see that the timezone is crucial at determining whether two dates are in the same day.
Calendar.current uses the local timezone of the device for almost everything it does. isDate(_:inSameDayAs:) is no exception. In your device's timezone, the two dates are not in the same day. However, when you print them out without a formatter, they are always printed in the UTC timezone. In the UTC timezone, they are in the same day, making you think Calendar.current is wrong. Assuming you actually want to see if the two dates are in the same day in your device's timezone, then Calendar.current is right, and you don't need to fix anything.
To print the two dates in your timezone, use a formatter:
let formatter = DateFormatter()
formatter.dateStyle = .long
formatter.timeStyle = .long
print(formatter.string(from: yourDate))
If you actually want to see if the two dates are in the same UTC day, then you can set the timezone of the Calendar:
var calendar = Calendar.current
calendar.timeZone = TimeZone(identifier: "UTC")!
// call calendar.isDate(_:_sameDayAs:) rather than using Calendar.current

Swift 4 wrong date and time

I am struggle with timezone because I tried to set time to midnight to 23:00:00, however, the result show start with 6am until 5am next day. I tried to set timezone to current, still same result. Here my code
let day1num = Int(todaynumber.string(from: Date()))! + 1
let day1start = Calendar.current.date(bySetting: .day, value: day1num, of: Date())
let day1end = Calendar.current.date(bySetting: .hour, value: 23, of: day1start!)
print("\(day1start!) to \(day1end!)")
I got result
2020-04-22 06:00:00 +0000 to 2020-04-23 05:00:00 +0000
I don't want 6am to 5am next day, I want result:
2020-04-22 00:00:00 +0000 to 2020-04-22 23:00:00 +0000
How can I solve it?
Thanks!
Your issue is you're misunderstanding Date and how it prints to the console.
Dates are representative of moments in time. Think of them as an Integer that represents the amount of time that has passed since a reference date. It's NOT a human-readable string. To convert a date to a human-readable string, you need to use a DateFormatter and make sure to set the timeZone of the date formatter to be the time zone in which you want the string to be representative of (it will default to the system's current time zone). The timeZone will impact what the resulting string is. For example, if you have a date formatter where the time zone is Pacific time, it might return a value like April 22, 2020 6:00 PM, but then if you change the time zone to mountain time and get the string from the date, it will return April 22, 2020 7:00 PM.
When you print a Date instance to the console, the system formats it to be a human-readable string in the UTC time zone. If I'm doing my math correctly, you're in mountain time, which is why the value you're seeing logged is 6 hours ahead of the value you're expecting.
If you want to see the date logged to the console as the user will see it, you should use a DateFormatter instance. It will default to use the user's system time zone, then use the Date value you've calculated to get the String representation of that date, and then log that String to the console, rather than the date itself.
let formatter = DateFormatter()
formatter.dateStyle = .medium // Customize as needed
formatter.timeStyle = .medium // Customize as needed
// formatter.timeZone = ... something else if you don't want to use the system time zone
let day1StartAsString = formatter.string(from: day1start!)
let day1EndAsString = formatter.string(from: day1end!)
print("\(day1StartAsString) to \(day1EndAsString)")

Getting next date when trying to convert NSDate to String

I have this date in actuall
2016-09-03 19:00:00 +0000
Now I am trying to convert it to String using a specific format like below
But what I am getting in return is not as desired. the formatter is adding on day to the given date like below
Is this standard behaviour ?
This is not standard behaviour. This happen because of the time zone difference. Set time zone proper
Set the timezone.
formatter.timeZone = [NSTimeZone timeZoneWithAbbreviation: #"GMT"];
When you hover over the date, you can see that it is showing UTC, whereas the formatter is automatically converting this to a local date. If your timezone is 5 hours ahead of UTC, then it will be the next day locally from that time.

convert NSDate to GMT timezone

I am working on NSDate and i am new for it.I have start date and end date,getting from user.And it is 2013-01-01 and 2013-02-19.When i try to display in console it is showing me 2012-12-31 18:30:00 +0000.So you can say 5:30 is gap.So i am adding time interval
startDate = [aStartdDate dateByAddingTimeInterval:19800];
endDate = [aEndDate dateByAddingTimeInterval:19800];
My question is when i am trying to get current week using 2012-12-31 18:30:00 +0000 date, it is showing me correctly.But when i use 2013-02-01 00:00:00 +0000 date,it is showing me total number of weeks in month.Please help me.Thanking you.
Timezones shouldn't be part of your model - They are a presentation problem.
Your model should always use a common default timezone. If you present the date to your users, apply a NSDateFormatter that uses a specific timezone.
In your case this means, that you shouldn't try to fix your dates by applying arbitrary intervals but use a date formatter in the final step (output) instead.

Resources