mirror of
https://github.com/samsonjs/samhuri.net-ios.git
synced 2026-03-25 09:25:47 +00:00
438 lines
19 KiB
Objective-C
438 lines
19 KiB
Objective-C
//
|
|
// PostsViewController.m
|
|
// Blog
|
|
//
|
|
// Created by Sami Samhuri on 2014-10-18.
|
|
// Copyright (c) 2014 Guru Logic Inc. All rights reserved.
|
|
//
|
|
|
|
#import <PromiseKit/Promise.h>
|
|
#import <ObjectiveSugar/NSArray+ObjectiveSugar.h>
|
|
#import "PostsViewController.h"
|
|
#import "EditorViewController.h"
|
|
#import "Post.h"
|
|
#import "BlogController.h"
|
|
#import "PostCell.h"
|
|
#import "BlogStatus.h"
|
|
#import "NSDate+marshmallows.h"
|
|
#import "UIColor+Hex.h"
|
|
#import "PostCollection.h"
|
|
#import "ModelStore.h"
|
|
#import "UIImage+FontAwesome.h"
|
|
|
|
@interface PostsViewController ()
|
|
|
|
@property (nonatomic, strong) NSArray *postCollections;
|
|
@property (nonatomic, readonly, strong) NSMutableArray *drafts;
|
|
@property (nonatomic, readonly, strong) NSMutableArray *publishedPosts;
|
|
@property (nonatomic, strong) IBOutlet UIBarButtonItem *publishButton;
|
|
@property (nonatomic, strong) IBOutlet UIBarButtonItem *addButton;
|
|
@property (nonatomic, weak) UILabel *titleLabel;
|
|
@property (nonatomic, weak) UILabel *statusLabel;
|
|
@property (nonatomic, copy) NSString *blogStatusText;
|
|
@property (nonatomic, strong) NSDate *blogStatusDate;
|
|
@property (nonatomic, strong) NSTimer *blogStatusTimer;
|
|
@property (nonatomic, weak) NSLayoutConstraint *titleViewWidthConstraint;
|
|
@property (nonatomic, weak) NSLayoutConstraint *titleViewHeightConstraint;
|
|
@property (nonatomic, weak) NSLayoutConstraint *titleLabelTopConstraint;
|
|
|
|
@end
|
|
|
|
static const NSUInteger SectionDrafts = 0;
|
|
static const NSUInteger SectionPublished = 1;
|
|
|
|
@interface TitleView : UIView @end
|
|
@implementation TitleView
|
|
- (void)addConstraint:(NSLayoutConstraint *)constraint {
|
|
if (![@"NSAutoresizingMaskLayoutConstraint" isEqualToString:NSStringFromClass([constraint class])]) {
|
|
[super addConstraint:constraint];
|
|
}
|
|
}
|
|
@end
|
|
|
|
@implementation PostsViewController
|
|
|
|
@dynamic drafts, publishedPosts;
|
|
|
|
- (void)awakeFromNib {
|
|
[super awakeFromNib];
|
|
if ([[UIDevice currentDevice] userInterfaceIdiom] == UIUserInterfaceIdiomPad) {
|
|
self.clearsSelectionOnViewWillAppear = NO;
|
|
self.preferredContentSize = CGSizeMake(320.0, 600.0);
|
|
}
|
|
|
|
[self setupTitleView];
|
|
[self setupFontAwesomeIcons];
|
|
self.refreshControl.tintColor = [UIColor whiteColor];
|
|
}
|
|
|
|
- (void)setupTitleView {
|
|
TitleView *titleView = [[TitleView alloc] initWithFrame:CGRectZero];
|
|
titleView.translatesAutoresizingMaskIntoConstraints = NO;
|
|
NSLayoutConstraint *widthConstraint = [NSLayoutConstraint constraintWithItem:titleView attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationGreaterThanOrEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1 constant:0];
|
|
[titleView addConstraint:widthConstraint];
|
|
self.titleViewWidthConstraint = widthConstraint;
|
|
NSLayoutConstraint *heightConstraint = [NSLayoutConstraint constraintWithItem:titleView attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationGreaterThanOrEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1 constant:0];
|
|
[titleView addConstraint:heightConstraint];
|
|
self.titleViewHeightConstraint = heightConstraint;
|
|
titleView.translatesAutoresizingMaskIntoConstraints = NO;
|
|
titleView.clipsToBounds = YES;
|
|
titleView.userInteractionEnabled = YES;
|
|
UITapGestureRecognizer *recognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(requestStatusWithoutCaching)];
|
|
recognizer.numberOfTapsRequired = 2;
|
|
[titleView addGestureRecognizer:recognizer];
|
|
UILabel *titleLabel = [[UILabel alloc] initWithFrame:CGRectZero];
|
|
titleLabel.translatesAutoresizingMaskIntoConstraints = NO;
|
|
titleLabel.font = [UIFont fontWithName:@"MuseoSans-300" size:16];
|
|
titleLabel.textColor = [UIColor whiteColor];
|
|
titleLabel.text = self.navigationItem.title;
|
|
[titleLabel sizeToFit];
|
|
[titleView addSubview:titleLabel];
|
|
NSLayoutConstraint *topConstraint = [NSLayoutConstraint constraintWithItem:titleLabel attribute:NSLayoutAttributeTop relatedBy:NSLayoutRelationEqual toItem:titleView attribute:NSLayoutAttributeTop multiplier:1 constant:0];
|
|
[titleView addConstraint:topConstraint];
|
|
self.titleLabelTopConstraint = topConstraint;
|
|
[titleView addConstraint:[NSLayoutConstraint constraintWithItem:titleLabel attribute:NSLayoutAttributeCenterX relatedBy:NSLayoutRelationEqual toItem:titleView attribute:NSLayoutAttributeCenterX multiplier:1 constant:0]];
|
|
self.titleLabel = titleLabel;
|
|
UILabel *subtitleLabel = [[UILabel alloc] initWithFrame:CGRectZero];
|
|
subtitleLabel.translatesAutoresizingMaskIntoConstraints = NO;
|
|
subtitleLabel.font = [UIFont systemFontOfSize:11];
|
|
subtitleLabel.textColor = [UIColor whiteColor];
|
|
[subtitleLabel sizeToFit];
|
|
[titleView addSubview:subtitleLabel];
|
|
[titleView addConstraint:[NSLayoutConstraint constraintWithItem:subtitleLabel attribute:NSLayoutAttributeBottom relatedBy:NSLayoutRelationEqual toItem:titleView attribute:NSLayoutAttributeBottom multiplier:1 constant:-9]];
|
|
[titleView addConstraint:[NSLayoutConstraint constraintWithItem:subtitleLabel attribute:NSLayoutAttributeCenterX relatedBy:NSLayoutRelationEqual toItem:titleView attribute:NSLayoutAttributeCenterX multiplier:1 constant:0]];
|
|
self.statusLabel = subtitleLabel;
|
|
self.navigationItem.titleView = titleView;
|
|
}
|
|
|
|
- (void)updateTitleViewConstraints;
|
|
{
|
|
self.titleViewWidthConstraint.constant = CGRectGetWidth(self.view.bounds);
|
|
CGFloat height = CGRectGetHeight(self.navigationController.navigationBar.bounds);
|
|
CGFloat top = 5;
|
|
if (UIDeviceOrientationIsPortrait([UIDevice currentDevice].orientation))
|
|
{
|
|
// status bar
|
|
height += 20;
|
|
top += 15;
|
|
}
|
|
self.titleViewHeightConstraint.constant = height;
|
|
self.titleLabelTopConstraint.constant = top;
|
|
[self.titleLabel.superview setNeedsUpdateConstraints];
|
|
}
|
|
|
|
- (void)updateOnClassInjection {
|
|
[self.titleLabel.constraints each:^(NSLayoutConstraint *constraint) {
|
|
[constraint.secondItem removeConstraint:constraint];
|
|
}];
|
|
[self.statusLabel.constraints each:^(NSLayoutConstraint *constraint) {
|
|
[constraint.secondItem removeConstraint:constraint];
|
|
}];
|
|
[self setupTitleView];
|
|
}
|
|
|
|
- (void)setupFontAwesomeIcons {
|
|
UIImage *image = [UIImage imageWithIcon:@"fa-rss" backgroundColor:[UIColor clearColor] iconColor:[UIColor mm_colorFromInteger:0xAA0000] fontSize:20];
|
|
UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
|
|
[button setImage:image forState:UIControlStateNormal];
|
|
[button addTarget:self action:@selector(publish:) forControlEvents:UIControlEventTouchUpInside];
|
|
button.frame = CGRectMake(0, 0, image.size.width, image.size.height);
|
|
self.publishButton.customView = button;
|
|
}
|
|
|
|
- (void)setupBlogStatusTimer {
|
|
self.blogStatusTimer = [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(updateBlogStatus) userInfo:nil repeats:YES];
|
|
}
|
|
|
|
- (void)teardownBlogStatusTimer {
|
|
[self.blogStatusTimer invalidate];
|
|
self.blogStatusTimer = nil;
|
|
}
|
|
|
|
- (void)updateStatusLabel:(NSString *)blogStatus animated:(BOOL)animated {
|
|
if (self.statusLabel && ![self.statusLabel.text isEqualToString:blogStatus]) {
|
|
self.statusLabel.text = blogStatus;
|
|
[self.statusLabel sizeToFit];
|
|
UIView *titleView = self.statusLabel.superview;
|
|
CGFloat x = CGRectGetWidth(titleView.bounds) / 2;
|
|
CGFloat y = 50 + CGRectGetHeight(self.statusLabel.frame) / 2;
|
|
self.statusLabel.center = CGPointMake(x, y);
|
|
self.statusLabel.alpha = 0;
|
|
void (^animate)() = ^{
|
|
CGRect frame = self.statusLabel.frame;
|
|
frame.origin.y = CGRectGetMaxY(self.titleLabel.frame) + 3;
|
|
self.statusLabel.frame = frame;
|
|
self.statusLabel.alpha = 1;
|
|
};
|
|
if (animated) {
|
|
[UIView animateWithDuration:0.3 animations:animate];
|
|
}
|
|
else {
|
|
animate();
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)updateBlogStatus {
|
|
[self updateStatusLabel:[NSString stringWithFormat:@"%@ as of %@", self.blogStatusText, [self.blogStatusDate mm_relativeToNow]] animated:NO];
|
|
}
|
|
|
|
- (void)updateBlogStatusAnimated:(BOOL)animated {
|
|
[self updateStatusLabel:[NSString stringWithFormat:@"%@ as of %@", self.blogStatusText, [self.blogStatusDate mm_relativeToNow]] animated:animated];
|
|
}
|
|
|
|
- (void)viewWillAppear:(BOOL)animated {
|
|
[super viewWillAppear:animated];
|
|
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(postUpdated:) name:PostUpdatedNotification object:nil];
|
|
[self setupBlogStatusTimer];
|
|
[self requestStatusWithCaching:YES];
|
|
if (!self.postCollections) {
|
|
[self requestPostsWithCaching:YES];
|
|
}
|
|
if (self.tableView.indexPathForSelectedRow) {
|
|
[self.tableView deselectRowAtIndexPath:self.tableView.indexPathForSelectedRow animated:YES];
|
|
}
|
|
}
|
|
|
|
- (void)viewWillDisappear:(BOOL)animated {
|
|
[super viewWillDisappear:animated];
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self name:PostUpdatedNotification object:nil];
|
|
[self teardownBlogStatusTimer];
|
|
}
|
|
|
|
- (void)viewDidLayoutSubviews {
|
|
[super viewDidLayoutSubviews];
|
|
[self updateTitleViewConstraints];
|
|
}
|
|
|
|
- (IBAction)refresh:(id)sender {
|
|
[self requestStatusWithCaching:NO];
|
|
[self requestPostsWithCaching:NO].finally(^{
|
|
[self.refreshControl endRefreshing];
|
|
});
|
|
}
|
|
|
|
- (PMKPromise *)requestStatusWithoutCaching {
|
|
return [self requestStatusWithCaching:NO];
|
|
}
|
|
|
|
- (PMKPromise *)requestStatusWithCaching:(BOOL)useCache {
|
|
[self teardownBlogStatusTimer];
|
|
[self updateStatusLabel:@"Checking status" animated:YES];
|
|
return [self.blogController requestBlogStatusWithCaching:useCache].then(^(BlogStatus *status) {
|
|
self.blogStatusDate = status.date;
|
|
if (status.dirty) {
|
|
self.blogStatusText = @"Dirty";
|
|
}
|
|
else {
|
|
self.blogStatusText = @"Everything published";
|
|
}
|
|
[self setupBlogStatusTimer];
|
|
[self updateBlogStatusAnimated:YES];
|
|
return status;
|
|
}).catch(^(NSError *error) {
|
|
[self updateStatusLabel:@"Failed to check status" animated:NO];
|
|
return error;
|
|
});
|
|
}
|
|
|
|
- (PMKPromise *)requestPostsWithCaching:(BOOL)useCache {
|
|
return [self.blogController requestAllPostsWithCaching:useCache].then(^(NSArray *results) {
|
|
self.postCollections = @[
|
|
[PostCollection postCollectionWithTitle:@"Drafts" posts:results.firstObject],
|
|
[PostCollection postCollectionWithTitle:@"Published" posts:results.lastObject],
|
|
];
|
|
[self.tableView reloadData];
|
|
return results;
|
|
});
|
|
}
|
|
|
|
- (PostCollection *)postCollectionForSection:(NSInteger)section {
|
|
return self.postCollections[section];
|
|
}
|
|
|
|
- (Post *)postForIndexPath:(NSIndexPath *)indexPath {
|
|
return [self postCollectionForSection:indexPath.section].posts[indexPath.row];
|
|
}
|
|
|
|
- (NSMutableArray *)drafts {
|
|
return [self postCollectionForSection:SectionDrafts].posts;
|
|
}
|
|
|
|
- (NSMutableArray *)publishedPosts {
|
|
return [self postCollectionForSection:SectionPublished].posts;
|
|
}
|
|
|
|
- (void)didReceiveMemoryWarning {
|
|
[super didReceiveMemoryWarning];
|
|
// Dispose of any resources that can be recreated.
|
|
}
|
|
|
|
- (IBAction)insertNewObject:(id)sender {
|
|
NSString *title = [UIPasteboard generalPasteboard].string;
|
|
NSURL *url = [UIPasteboard generalPasteboard].URL;
|
|
// TODO: image, anything else interesting
|
|
Post *post = [Post newDraftWithTitle:title body:nil url:url];
|
|
[self.drafts insertObject:post atIndex:0];
|
|
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:0 inSection:SectionDrafts];
|
|
[self.tableView insertRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
|
|
[self.tableView selectRowAtIndexPath:indexPath animated:YES scrollPosition:UITableViewScrollPositionTop];
|
|
[self performSegueWithIdentifier:@"showDetail" sender:sender];
|
|
}
|
|
|
|
- (IBAction)publish:(id)sender {
|
|
// TODO: activity indicator
|
|
__weak typeof(self) welf = self;
|
|
void (^publish)(PMKPromise *) = ^(PMKPromise *promise) {
|
|
self.publishButton.enabled = NO;
|
|
typeof(self) self = welf;
|
|
promise.then(^{
|
|
[self requestStatusWithoutCaching];
|
|
}).catch(^(NSError *error) {
|
|
UIAlertController *alertController = [UIAlertController alertControllerWithTitle:@"Error" message:error.localizedDescription preferredStyle:UIAlertControllerStyleAlert];
|
|
__weak typeof(self) welf = self;
|
|
[alertController addAction:[UIAlertAction actionWithTitle:@"Dismiss" style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {
|
|
typeof(self) self = welf;
|
|
[self dismissViewControllerAnimated:YES completion:nil];
|
|
}]];
|
|
[self presentViewController:alertController animated:YES completion:nil];
|
|
}).finally(^{
|
|
self.publishButton.enabled = YES;
|
|
});
|
|
};
|
|
UIAlertController *alertController = [UIAlertController alertControllerWithTitle:@"Publish" message:nil preferredStyle:UIAlertControllerStyleActionSheet];
|
|
[alertController addAction:[UIAlertAction actionWithTitle:@"samhuri.net" style:UIAlertActionStyleDestructive handler:^(UIAlertAction *action) {
|
|
typeof(self) self = welf;
|
|
[self dismissViewControllerAnimated:YES completion:nil];
|
|
publish([self.blogController requestPublishToProductionEnvironment]);
|
|
}]];
|
|
[alertController addAction:[UIAlertAction actionWithTitle:@"beta.samhuri.net" style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {
|
|
typeof(self) self = welf;
|
|
[self dismissViewControllerAnimated:YES completion:nil];
|
|
publish([self.blogController requestPublishToStagingEnvironment]);
|
|
}]];
|
|
[alertController addAction:[UIAlertAction actionWithTitle:@"Push to GitHub" style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {
|
|
typeof(self) self = welf;
|
|
[self dismissViewControllerAnimated:YES completion:nil];
|
|
publish([self.blogController requestSync]);
|
|
}]];
|
|
[alertController addAction:[UIAlertAction actionWithTitle:@"Cancel" style:UIAlertActionStyleCancel handler:^(UIAlertAction *action) {
|
|
typeof(self) self = welf;
|
|
[self dismissViewControllerAnimated:YES completion:nil];
|
|
}]];
|
|
[self presentViewController:alertController animated:YES completion:nil];
|
|
}
|
|
|
|
- (void)postUpdated:(NSNotification *)note {
|
|
Post *post = note.userInfo[PostUserInfoKey];
|
|
BOOL (^isThisPost)(Post *, NSUInteger, BOOL *) = ^BOOL(Post *p, NSUInteger idx, BOOL *stop) {
|
|
return [p.objectID isEqualToString:post.objectID];
|
|
};
|
|
NSUInteger section = SectionDrafts;
|
|
NSUInteger row = [self.drafts indexOfObjectPassingTest:isThisPost];
|
|
if (row == NSNotFound) {
|
|
section = SectionPublished;
|
|
row = [self.publishedPosts indexOfObjectPassingTest:isThisPost];
|
|
}
|
|
if (row != NSNotFound) {
|
|
PostCollection *collection = [self postCollectionForSection:section];
|
|
[collection.posts replaceObjectAtIndex:row withObject:post];
|
|
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:row inSection:section];
|
|
[self.tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
|
|
}
|
|
}
|
|
|
|
#pragma mark - Segues
|
|
|
|
- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender {
|
|
if ([segue.identifier isEqualToString:@"showDetail"]) {
|
|
NSIndexPath *indexPath = [self.tableView indexPathForSelectedRow];
|
|
Post *post = [self postForIndexPath:indexPath];
|
|
EditorViewController *controller = (EditorViewController *)[[segue destinationViewController] topViewController];
|
|
controller.blogController = self.blogController;
|
|
[controller configureWithPost:post];
|
|
controller.navigationItem.leftBarButtonItem = self.splitViewController.displayModeButtonItem;
|
|
controller.navigationItem.leftItemsSupplementBackButton = YES;
|
|
}
|
|
}
|
|
|
|
#pragma mark - State restoration
|
|
|
|
static NSString *const StateRestorationPostCollectionsKey = @"postCollections";
|
|
static NSString *const StateRestorationBlogStatusDateKey = @"blogStatusDate";
|
|
static NSString *const StateRestorationBlogStatusTextKey = @"blogStatusText";
|
|
|
|
- (void)encodeRestorableStateWithCoder:(NSCoder *)coder {
|
|
NSLog(@"%@ encode restorable state with coder %@", self, coder);
|
|
[coder encodeObject:self.postCollections forKey:StateRestorationPostCollectionsKey];
|
|
[coder encodeObject:self.blogStatusDate forKey:StateRestorationBlogStatusDateKey];
|
|
[coder encodeObject:self.blogStatusText forKey:StateRestorationBlogStatusTextKey];
|
|
[super encodeRestorableStateWithCoder:coder];
|
|
}
|
|
|
|
- (void)decodeRestorableStateWithCoder:(NSCoder *)coder {
|
|
NSLog(@"%@ decode restorable state with coder %@", self, coder);
|
|
self.postCollections = [coder decodeObjectForKey:StateRestorationPostCollectionsKey];
|
|
self.blogStatusDate = [coder decodeObjectForKey:StateRestorationBlogStatusDateKey];
|
|
self.blogStatusText = [coder decodeObjectForKey:StateRestorationBlogStatusTextKey];
|
|
[super decodeRestorableStateWithCoder:coder];
|
|
}
|
|
|
|
#pragma mark - Table View
|
|
|
|
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
|
|
return self.postCollections.count;
|
|
}
|
|
|
|
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
|
|
return [self postCollectionForSection:section].posts.count;
|
|
}
|
|
|
|
- (NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section {
|
|
return [self postCollectionForSection:section].title;
|
|
}
|
|
|
|
- (CGFloat)tableView:(UITableView *)tableView heightForHeaderInSection:(NSInteger)section {
|
|
return 40;
|
|
}
|
|
|
|
- (CGFloat)tableView:(UITableView *)tableView heightForFooterInSection:(NSInteger)section {
|
|
return 0.01;
|
|
}
|
|
|
|
- (void)tableView:(UITableView *)tableView willDisplayHeaderView:(UIView *)view forSection:(NSInteger)section {
|
|
UITableViewHeaderFooterView *headerView = [view isKindOfClass:[UITableViewHeaderFooterView class]] ? (UITableViewHeaderFooterView *)view : nil;
|
|
headerView.textLabel.textColor = [UIColor mm_colorFromInteger:0xF7F7F7];
|
|
}
|
|
|
|
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
|
|
PostCell *cell = [tableView dequeueReusableCellWithIdentifier:@"Cell" forIndexPath:indexPath];
|
|
Post *post = [self postForIndexPath:indexPath];
|
|
NSString *title = post.title.length ? post.title : @"Untitled";
|
|
NSString *date = post.draft ? @"" : post.formattedDate;
|
|
[cell configureWithTitle:title date:date];
|
|
return cell;
|
|
}
|
|
|
|
- (BOOL)tableView:(UITableView *)tableView canEditRowAtIndexPath:(NSIndexPath *)indexPath {
|
|
// Return NO if you do not want the specified item to be editable.
|
|
return YES;
|
|
}
|
|
|
|
- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath {
|
|
if (editingStyle == UITableViewCellEditingStyleDelete) {
|
|
PostCollection *collection = [self postCollectionForSection:indexPath.section];
|
|
Post *post = [self postForIndexPath:indexPath];
|
|
// TODO: activity indicator
|
|
[self.blogController requestDeletePost:post].then(^{
|
|
[collection.posts removeObjectAtIndex:indexPath.row];
|
|
[tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationFade];
|
|
});
|
|
}
|
|
else if (editingStyle == UITableViewCellEditingStyleInsert) {
|
|
// TODO: determine when this is called and see if we actually need it
|
|
// Create a new instance of the appropriate class, insert it into the array, and add a new row to the table view.
|
|
}
|
|
}
|
|
|
|
@end
|