1996-05-30 20:03:15 +00:00
|
|
|
/*
|
|
|
|
NSImageRep.m
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
Abstract representation of an image.
|
1996-05-30 20:03:15 +00:00
|
|
|
|
|
|
|
Copyright (C) 1996 Free Software Foundation, Inc.
|
|
|
|
|
1996-10-18 17:14:13 +00:00
|
|
|
Author: Adam Fedor <fedor@colorado.edu>
|
1996-08-22 18:51:08 +00:00
|
|
|
Date: Feb 1996
|
|
|
|
|
|
|
|
This file is part of the GNUstep Application Kit Library.
|
1996-05-30 20:03:15 +00:00
|
|
|
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU Library General Public
|
|
|
|
License as published by the Free Software Foundation; either
|
|
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
Library General Public License for more details.
|
1996-08-22 18:51:08 +00:00
|
|
|
|
1996-05-30 20:03:15 +00:00
|
|
|
You should have received a copy of the GNU Library General Public
|
1996-10-18 17:14:13 +00:00
|
|
|
License along with this library; see the file COPYING.LIB.
|
|
|
|
If not, write to the Free Software Foundation,
|
|
|
|
59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
1996-05-30 20:03:15 +00:00
|
|
|
*/
|
|
|
|
|
1997-09-23 22:43:24 +00:00
|
|
|
#include <gnustep/gui/config.h>
|
1996-08-22 18:51:08 +00:00
|
|
|
#include <string.h>
|
|
|
|
#include <Foundation/NSArray.h>
|
|
|
|
#include <Foundation/NSData.h>
|
|
|
|
#include <Foundation/NSNotification.h>
|
|
|
|
#include <AppKit/NSImageRep.h>
|
|
|
|
#include <AppKit/NSBitmapImageRep.h>
|
|
|
|
#include <AppKit/NSEPSImageRep.h>
|
1997-02-18 00:29:25 +00:00
|
|
|
#include <AppKit/NSPasteboard.h>
|
1996-05-30 20:03:15 +00:00
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
/* Backend protocol - methods that must be implemented by the backend to
|
|
|
|
complete the class */
|
|
|
|
@protocol NXImageRepBackend
|
|
|
|
- (BOOL) drawAtPoint: (NSPoint)aPoint;
|
|
|
|
- (BOOL) drawInRect: (NSRect)aRect;
|
|
|
|
@end
|
|
|
|
|
|
|
|
static NSMutableArray* imageReps = NULL;
|
|
|
|
|
|
|
|
/* Get the extension from a name */
|
|
|
|
static NSString *
|
|
|
|
extension(NSString *name)
|
|
|
|
{
|
|
|
|
/* Waiting for NSString to be complete */
|
|
|
|
#if 0
|
|
|
|
return [name pathExtension];
|
|
|
|
#else
|
1996-09-08 14:32:34 +00:00
|
|
|
const char* cname;
|
1996-08-22 18:51:08 +00:00
|
|
|
char *s;
|
1996-09-08 14:32:34 +00:00
|
|
|
|
|
|
|
cname = [name cString];
|
|
|
|
s = strrchr(cname, '.');
|
|
|
|
if (s > strrchr(cname, '/'))
|
1996-08-22 18:51:08 +00:00
|
|
|
return [NSString stringWithCString:s+1];
|
|
|
|
else
|
|
|
|
return nil;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
1996-05-30 20:03:15 +00:00
|
|
|
@implementation NSImageRep
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (void) initialize
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
/* While there are four imageRep subclasses, in practice, only two of
|
|
|
|
them can load in data from an external source. */
|
1996-09-08 14:32:34 +00:00
|
|
|
if (self == [NSImageRep class])
|
|
|
|
{
|
|
|
|
imageReps = [[NSMutableArray alloc] initWithCapacity: 2];
|
1996-10-18 17:14:13 +00:00
|
|
|
// [imageReps addObject: [NSBitmapImageRep class]];
|
|
|
|
// [imageReps addObject: [NSEPSImageRep class]];
|
1996-09-08 14:32:34 +00:00
|
|
|
}
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Creating an NSImageRep
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (id) imageRepWithContentsOfFile: (NSString *)filename
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
NSArray* array;
|
|
|
|
|
|
|
|
array = [self imageRepsWithContentsOfFile: filename];
|
|
|
|
if ([array count])
|
|
|
|
return [array objectAtIndex: 0];
|
1996-05-30 20:03:15 +00:00
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (NSArray *) imageRepsWithContentsOfFile: (NSString *)filename
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
int i, count;
|
|
|
|
NSString* ext;
|
|
|
|
NSMutableArray* array;
|
|
|
|
|
|
|
|
ext = extension(filename);
|
|
|
|
// FIXME: Should this be an exception? Should we even check this?
|
|
|
|
if (!ext)
|
|
|
|
return nil;
|
|
|
|
array = [NSMutableArray arrayWithCapacity:1];
|
|
|
|
|
|
|
|
count = [imageReps count];
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
{
|
|
|
|
Class rep = [imageReps objectAtIndex: i];
|
1996-10-18 17:14:13 +00:00
|
|
|
#if 0
|
1996-08-22 18:51:08 +00:00
|
|
|
if ([[rep imageFileTypes] indexOfObject: ext] != NSNotFound)
|
1996-09-08 14:32:34 +00:00
|
|
|
#else
|
|
|
|
/* xxxFIXME: not implemented in gcc-2.7.2 runtime. */
|
|
|
|
if ([rep respondsToSelector: @selector(imageFileTypes)]
|
|
|
|
&& [[rep imageFileTypes] indexOfObject: ext] != NSNotFound)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
NSData* data = [NSData dataWithContentsOfFile: filename];
|
1996-10-18 17:14:13 +00:00
|
|
|
#if 1
|
1996-09-08 14:32:34 +00:00
|
|
|
if ([rep respondsToSelector: @selector(imageRepsWithData:)])
|
|
|
|
#endif
|
|
|
|
[array addObjectsFromArray: [rep imageRepsWithData: data]];
|
1996-10-18 17:14:13 +00:00
|
|
|
#if 1
|
1996-09-08 14:32:34 +00:00
|
|
|
else if ([rep respondsToSelector: @selector(imageRepWithData:)])
|
|
|
|
[array addObject: [rep imageRepWithData: data]];
|
1996-08-22 18:51:08 +00:00
|
|
|
#endif
|
1996-09-08 14:32:34 +00:00
|
|
|
}
|
1996-08-22 18:51:08 +00:00
|
|
|
}
|
|
|
|
return (NSArray *)array;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (id) imageRepWithPasteboard: (NSPasteboard *)pasteboard
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
NSArray* array;
|
|
|
|
|
|
|
|
array = [self imageRepsWithPasteboard: pasteboard];
|
|
|
|
if ([array count])
|
|
|
|
return [array objectAtIndex: 0];
|
1996-05-30 20:03:15 +00:00
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (NSArray *) imageRepsWithPasteboard: (NSPasteboard *)pasteboard
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
int i, count;
|
|
|
|
NSMutableArray* array;
|
|
|
|
|
|
|
|
array = [NSMutableArray arrayWithCapacity:1];
|
|
|
|
|
|
|
|
count = [imageReps count];
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
{
|
|
|
|
NSString* ptype;
|
|
|
|
Class rep = [imageReps objectAtIndex: i];
|
|
|
|
if ([rep respondsToSelector: @selector(imagePasteboardTypes)]
|
|
|
|
&& (ptype =
|
|
|
|
[pasteboard availableTypeFromArray:[rep imagePasteboardTypes]]))
|
|
|
|
{
|
|
|
|
NSData* data = [pasteboard dataForType: ptype];
|
|
|
|
if ([rep respondsToSelector: @selector(imageRepsWithData:)])
|
|
|
|
[array addObjectsFromArray: [rep imageRepsWithData: data]];
|
|
|
|
else if ([rep respondsToSelector: @selector(imageRepWithData:)])
|
|
|
|
[array addObject: [rep imageRepWithData: data]];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (NSArray *)array;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void) dealloc
|
|
|
|
{
|
1996-09-08 14:32:34 +00:00
|
|
|
[_colorSpace release];
|
1996-08-22 18:51:08 +00:00
|
|
|
[super dealloc];
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Checking Data Types
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (BOOL) canInitWithData: (NSData *)data
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
/* Subclass responsibility */
|
1996-05-30 20:03:15 +00:00
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (BOOL) canInitWithPasteboard: (NSPasteboard *)pasteboard
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
/* Subclass responsibility */
|
1996-05-30 20:03:15 +00:00
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (NSArray *) imageFileTypes
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
/* Subclass responsibility */
|
1996-05-30 20:03:15 +00:00
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (NSArray *) imagePasteboardTypes
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
/* Subclass responsibility */
|
1996-05-30 20:03:15 +00:00
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (NSArray *) imageUnfilteredFileTypes
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
/* Subclass responsibility */
|
1996-05-30 20:03:15 +00:00
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (NSArray *) imageUnfilteredPasteboardTypes
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
/* Subclass responsibility */
|
1996-05-30 20:03:15 +00:00
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
// Setting the Size of the Image
|
|
|
|
- (void) setSize: (NSSize)aSize
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
size = aSize;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (NSSize) size
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
return size;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
// Specifying Information about the Representation
|
|
|
|
- (int) bitsPerSample
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
return bitsPerSample;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (NSString *) colorSpaceName
|
|
|
|
{
|
1996-09-08 14:32:34 +00:00
|
|
|
return _colorSpace;
|
1996-08-22 18:51:08 +00:00
|
|
|
}
|
1996-05-30 20:03:15 +00:00
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (BOOL) hasAlpha
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
return hasAlpha;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (BOOL) isOpaque
|
|
|
|
{
|
|
|
|
return isOpaque;
|
|
|
|
}
|
1996-05-30 20:03:15 +00:00
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (int) pixelsWide
|
|
|
|
{
|
|
|
|
return _pixelsWide;
|
|
|
|
}
|
1996-05-30 20:03:15 +00:00
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (int) pixelsHigh
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
return _pixelsHigh;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (void) setAlpha: (BOOL)flag
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
hasAlpha = flag;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (void) setBitsPerSample: (int)anInt
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
bitsPerSample = anInt;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (void) setColorSpaceName: (NSString *)aString
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-09-08 14:32:34 +00:00
|
|
|
[_colorSpace autorelease];
|
|
|
|
_colorSpace = [aString retain];
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (void) setOpaque: (BOOL)flag
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
isOpaque = flag;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (void) setPixelsWide: (int)anInt
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
_pixelsWide = anInt;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (void) setPixelsHigh: (int)anInt
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
_pixelsHigh = anInt;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
// Drawing the Image
|
|
|
|
- (BOOL) draw
|
|
|
|
{
|
|
|
|
[self subclassResponsibility: _cmd];
|
|
|
|
return NO;
|
|
|
|
}
|
1996-05-30 20:03:15 +00:00
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (BOOL) drawAtPoint: (NSPoint)aPoint
|
|
|
|
{
|
|
|
|
return NO;
|
|
|
|
}
|
1996-05-30 20:03:15 +00:00
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- (BOOL) drawInRect: (NSRect)aRect
|
|
|
|
{
|
|
|
|
return NO;
|
|
|
|
}
|
1996-05-30 20:03:15 +00:00
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
// Managing NSImageRep Subclasses
|
|
|
|
+ (Class) imageRepClassForData: (NSData *)data
|
|
|
|
{
|
|
|
|
int i, count;
|
1996-05-30 20:03:15 +00:00
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
count = [imageReps count];
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
{
|
|
|
|
Class rep = [imageReps objectAtIndex: i];
|
|
|
|
if ([rep canInitWithData: data])
|
|
|
|
return rep;
|
|
|
|
}
|
|
|
|
return Nil;
|
|
|
|
}
|
1996-05-30 20:03:15 +00:00
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (Class) imageRepClassForFileType: (NSString *)type
|
|
|
|
{
|
|
|
|
int i, count;
|
1996-05-30 20:03:15 +00:00
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
count = [imageReps count];
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
{
|
|
|
|
Class rep = [imageReps objectAtIndex: i];
|
|
|
|
if ([rep respondsToSelector: @selector(imageFileTypes)]
|
|
|
|
&& [[rep imageFileTypes] indexOfObject: type] != NSNotFound)
|
|
|
|
{
|
|
|
|
return rep;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (Class) imageRepClassForPasteboardType: (NSString *)type
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
int i, count;
|
|
|
|
|
|
|
|
count = [imageReps count];
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
{
|
|
|
|
Class rep = [imageReps objectAtIndex: i];
|
|
|
|
if ([rep respondsToSelector: @selector(imagePasteboardTypes)]
|
|
|
|
&& ([[rep imagePasteboardTypes] indexOfObject: type] != NSNotFound))
|
|
|
|
{
|
|
|
|
return rep;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Nil;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (void) registerImageRepClass: (Class)imageRepClass
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
[imageReps addObject: imageRepClass];
|
1996-09-08 14:32:34 +00:00
|
|
|
/*
|
1996-08-22 18:51:08 +00:00
|
|
|
[[NSNotificationCenter defaultCenter]
|
|
|
|
postNotificationName: NSImageRepRegistryChangedNotification
|
|
|
|
object: self];
|
1996-09-08 14:32:34 +00:00
|
|
|
*/
|
1996-08-22 18:51:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
+ (NSArray *) registeredImageRepClasses
|
|
|
|
{
|
|
|
|
return (NSArray *)imageReps;
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
+ (void) unregisterImageRepClass: (Class)imageRepClass
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-08-22 18:51:08 +00:00
|
|
|
[imageReps removeObject: imageRepClass];
|
|
|
|
[[NSNotificationCenter defaultCenter]
|
|
|
|
postNotificationName: NSImageRepRegistryChangedNotification
|
|
|
|
object: self];
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NSCoding protocol
|
1996-08-22 18:51:08 +00:00
|
|
|
- (void) encodeWithCoder: aCoder
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-09-08 14:32:34 +00:00
|
|
|
[aCoder encodeObject: _colorSpace];
|
1996-08-22 18:51:08 +00:00
|
|
|
[aCoder encodeSize: size];
|
|
|
|
[aCoder encodeValueOfObjCType: @encode(BOOL) at: &hasAlpha];
|
|
|
|
[aCoder encodeValueOfObjCType: @encode(BOOL) at: &isOpaque];
|
|
|
|
[aCoder encodeValueOfObjCType: @encode(int) at: &bitsPerSample];
|
|
|
|
[aCoder encodeValueOfObjCType: @encode(int) at: &_pixelsWide];
|
|
|
|
[aCoder encodeValueOfObjCType: @encode(int) at: &_pixelsHigh];
|
1996-05-30 20:03:15 +00:00
|
|
|
}
|
|
|
|
|
1996-08-22 18:51:08 +00:00
|
|
|
- initWithCoder: aDecoder
|
1996-05-30 20:03:15 +00:00
|
|
|
{
|
1996-09-08 14:32:34 +00:00
|
|
|
_colorSpace = [[aDecoder decodeObject] retain];
|
1996-08-22 18:51:08 +00:00
|
|
|
size = [aDecoder decodeSize];
|
|
|
|
[aDecoder decodeValueOfObjCType: @encode(BOOL) at: &hasAlpha];
|
|
|
|
[aDecoder decodeValueOfObjCType: @encode(BOOL) at: &isOpaque];
|
|
|
|
[aDecoder decodeValueOfObjCType: @encode(int) at: &bitsPerSample];
|
|
|
|
[aDecoder decodeValueOfObjCType: @encode(int) at: &_pixelsWide];
|
|
|
|
[aDecoder decodeValueOfObjCType: @encode(int) at: &_pixelsHigh];
|
1996-05-30 20:03:15 +00:00
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
1996-08-22 18:51:08 +00:00
|
|
|
|