Three20 : how to pass a class of objects between 2 views

前端 未结 4 1004
长情又很酷
长情又很酷 2021-02-04 17:18

I have a TTableView. The items in this table a mapped to an url, so that when I click on an item, another view appear with informations about this item. All these informations a

相关标签:
4条回答
  • 2021-02-04 17:37

    I have little fix this awesome code :) as posted version strip callback from TTTableButton.

    Correction is:

    if( [object isKindOfClass:[TTTableButton class]] ) {
       if (item.delegate && item.selector) {
           [item.delegate performSelector:item.selector withObject:object];
       }
       [tableView deselectRowAtIndexPath:indexPath animated:YES];
    }
    
    0 讨论(0)
  • 2021-02-04 17:46

    The big problem with directly using TTURLAction is that you can't really use them with TTTableViewItem. The only way to really do it is override -didSelectObject:atIndexPath: and build your custom TTURLAction with your desired object in the query dictionary. But this breaks the nice separation of Model and View Controller, and gets complicated once you have multiple objects to pass.

    Instead of this, I've been using a small category which automatically takes the userInfo property of the table item (which I set to whatever I need), and automatically adds it as a URL parameter.

    And then you use this to retrieve it in your mapped view controller.

    - (id)initWithNavigatorURL:(NSURL *)URL query:(NSDictionary *)query { 
    if (self = [self initWithNibName:nil bundle:nil]) {
        id myPassedObject = [query objectForKey:@"__userInfo__"];
        // do the rest of your initlization
    }
    return self;
    

    }

    You can download it as a GitHub Gist here. The code is also below. We're considering merging this into the main branch at some point.


    TTTableViewDelegate+URLAdditions.h

    @interface TTTableViewDelegate(URLAdditions)
    
    @end
    

    TTTableViewDelegate+URLAdditions.m

    #import "TTTableViewDelegate+URLAdditions.h"
    
    @implementation TTTableViewDelegate(URLAdditions)
    
    - (void)tableView:(UITableView*)tableView didSelectRowAtIndexPath:(NSIndexPath*)indexPath {
        id<TTTableViewDataSource> dataSource = (id<TTTableViewDataSource>)tableView.dataSource;
        id object = [dataSource tableView:tableView objectForRowAtIndexPath:indexPath];
    
        // Added section to automatically wrap up any TTTableItem userInfo objects.  If it is a dictionary, it gets sent directly
        // If it is not, it is put in a dictionary and sent as they __userInfo__ key
        if( [object isKindOfClass:[TTTableLinkedItem class]] ) {
            TTTableLinkedItem* item = object;
    
            if( item.URL && [_controller shouldOpenURL:item.URL] ) {
                // If the TTTableItem has userInfo, wrap it up and send it along to the URL
                if( item.userInfo ) {
                    NSDictionary *userInfoDict;
    
                    // If userInfo is a dictionary, pass it along else create a dictionary
                    if( [item.userInfo isKindOfClass:[NSDictionary class]] ) {
                        userInfoDict = item.userInfo;
                    } else {
                        userInfoDict = [NSDictionary dictionaryWithObject:item.userInfo forKey:@"__userInfo__"];
                    }
    
                    [[TTNavigator navigator] openURLAction:[[[TTURLAction actionWithURLPath:item.URL]
                                                             applyQuery:userInfoDict]
                                                            applyAnimated:YES]];
                } else {
                    TTOpenURL( item.URL );
                }
            }
    
            if( [object isKindOfClass:[TTTableButton class]] ) {
                [tableView deselectRowAtIndexPath:indexPath animated:YES];
            }
            else if( [object isKindOfClass:[TTTableMoreButton class]] ) {
                TTTableMoreButton* moreLink = (TTTableMoreButton*)object;
                moreLink.isLoading = YES;
                TTTableMoreButtonCell* cell
                = (TTTableMoreButtonCell*)[tableView cellForRowAtIndexPath:indexPath];
                cell.animating = YES;
                [tableView deselectRowAtIndexPath:indexPath animated:YES];
    
                if( moreLink.model )
                    [moreLink.model load:TTURLRequestCachePolicyDefault more:YES];
                else
                    [_controller.model load:TTURLRequestCachePolicyDefault more:YES];
            }
        }
    
        [_controller didSelectObject:object atIndexPath:indexPath];
    }
    
    @end
    
    0 讨论(0)
  • 2021-02-04 17:50

    I think the documentation on the official website does describe very clearly the navigation scheme for Three20. Your question is the very common task of any application, and Three20 provides powerful support for that.

    0 讨论(0)
  • 2021-02-04 18:02

    One way of doing it is to use a TTURLAction. When the user selects a row in your table, which will call your didSelectObject (of TTTableViewController) method, extract the object or set of objects you want to pass and build a TTURLAction like this:

    TTURLAction *action =  [[[TTURLAction actionWithURLPath:@"tt://showUser"] 
        applyQuery:[NSDictionary dictionaryWithObject:user forKey:@"kParameterUser"]]
                applyAnimated:YES];
    

    Then open the action:

    [[TTNavigator navigator] openURLAction:action];
    

    The controller you want to open as a result of this action should be registered in your TTURLMap and should have a constructor thus:

    - (id) initWithNavigatorURL:(NSURL*)URL query:(NSDictionary*)query {
        self = [super init];
        if (self != nil) {
            self.user = [query objectForKey:kParameterUser];
        }
        return self;
    }
    

    I tend to create categories on classes for objects I want to be able to open another controller and display themselves.

    0 讨论(0)
提交回复
热议问题