2010-08-16 12 views
31

के भीतर इंटरसेप्ट उद्देश्य-सी प्रतिनिधि संदेश मेरे पास UIScrollView का उप-वर्ग है जिसमें मुझे आंतरिक रूप से स्क्रॉलिंग व्यवहार का जवाब देने की आवश्यकता है। हालांकि, व्यू कंट्रोलर को अभी भी स्क्रॉलिंग प्रतिनिधि कॉलबैक सुनने की आवश्यकता होगी, इसलिए मैं अपने घटक के भीतर प्रतिनिधि को चोरी नहीं कर सकता।एक सबक्लास

क्या "प्रतिनिधि" नाम की संपत्ति रखने के लिए कोई तरीका है और बस इसके साथ भेजे गए संदेशों को सुनें, या फिर किसी भी तरह आंतरिक रूप से प्रतिनिधि संपत्ति को हाइजैक करें और कुछ कोड चलाने के बाद संदेशों को आगे बढ़ाएं?

उत्तर

-5

हां, लेकिन आपको the docs में प्रत्येक प्रतिनिधि विधि को ओवरराइड करना होगा। असल में, दूसरी प्रतिनिधि संपत्ति बनाएं और प्रतिनिधि प्रोटोकॉल को कार्यान्वित करें। जब आपके प्रतिनिधि तरीकों को बुलाया जाता है, तो अपने व्यवसाय का ख्याल रखें और उसके बाद उसी प्रतिनिधि को प्रतिनिधि प्रतिनिधि विधि से अपने दूसरे प्रतिनिधि पर कॉल करें जो अभी चलाया गया था। जैसे

- (void)scrollViewDidScroll:(UIScrollView *)scrollView { 
    // Do stuff here 
    if ([self.delegate2 respondsToSelector:@selector(scrollViewDidScroll:)]) { 
     [self.delegate2 scrollViewDidScroll:scrollView]; 
    } 
} 
+3

Uglyyyyyyyyy, लेकिन दुर्भाग्य से आवश्यक:/ –

+0

हाँ, एकमात्र अन्य विकल्प nsnotifications –

+0

हैलो का उपयोग करना है ... मैंने इसे अपने स्वयं के कोड पर लागू किया है, कुछ प्रतिनिधि तरीकों के साथ यूआईटीएक्स्टफिल्ड के लिए यह काम करता है, लेकिन दूसरों के साथ परिणामस्वरूप अज्ञात चयनकर्ता उदाहरण 0x4b21800 ' – speeder

85

सभी प्रतिनिधि विधियों को मैन्युअल रूप से ओवरराइड करने से बचने के लिए, आप संदेश अग्रेषण का उपयोग कर सकते हैं। इस प्रकार मैं सिर्फ एक ही बात एक मध्यवर्ती प्रॉक्सी वर्ग का उपयोग कर लागू किया:

MessageInterceptor.h

@interface MessageInterceptor : NSObject { 
    id receiver; 
    id middleMan; 
} 
@property (nonatomic, assign) id receiver; 
@property (nonatomic, assign) id middleMan; 
@end 

MessageInterceptor.m

@implementation MessageInterceptor 
@synthesize receiver; 
@synthesize middleMan; 

- (id)forwardingTargetForSelector:(SEL)aSelector { 
    if ([middleMan respondsToSelector:aSelector]) { return middleMan; } 
    if ([receiver respondsToSelector:aSelector]) { return receiver; } 
    return [super forwardingTargetForSelector:aSelector]; 
} 

- (BOOL)respondsToSelector:(SEL)aSelector { 
    if ([middleMan respondsToSelector:aSelector]) { return YES; } 
    if ([receiver respondsToSelector:aSelector]) { return YES; } 
    return [super respondsToSelector:aSelector]; 
} 

@end 

MyScrollView.h

#import "MessageInterceptor.h" 

@interface MyScrollView : UIScrollView { 
    MessageInterceptor * delegate_interceptor; 
    //... 
} 

//... 

@end 

MyScrollView.m (jhabbott करने के लिए, संपादित धन्यवाद के साथ):

@implementation MyScrollView 

- (id)delegate { return delegate_interceptor.receiver; } 

- (void)setDelegate:(id)newDelegate { 
    [super setDelegate:nil]; 
    [delegate_interceptor setReceiver:newDelegate]; 
    [super setDelegate:(id)delegate_interceptor]; 
} 

- (id)init* { 
    //... 
    delegate_interceptor = [[MessageInterceptor alloc] init]; 
    [delegate_interceptor setMiddleMan:self]; 
    [super setDelegate:(id)delegate_interceptor]; 
    //... 
} 

- (void)dealloc { 
    //... 
    [delegate_interceptor release]; 
    //... 
} 

// delegate method override: 
- (void)scrollViewDidScroll:(UIScrollView *)scrollView { 
    // 1. your custom code goes here 
    // 2. forward to the delegate as usual 
    if ([self.delegate respondsToSelector:@selector(scrollViewDidScroll:)]) { 
     [self.delegate scrollViewDidScroll:scrollView]; 
    } 
} 

@end 
इस दृष्टिकोण के साथ

, MessageInterceptor वस्तु स्वचालित रूप से आगे नियमित प्रतिनिधि वस्तु के लिए सभी प्रतिनिधि संदेश, लोगों के लिए सिवाय इसके कि आप अपने कस्टम उपclass में ओवरराइड करते हैं।

+5

यह एक बहुत ही सुरुचिपूर्ण समाधान है और मैंने इसे दो बार उपयोग किया है। बहुत अच्छा। –

+0

@ सिमॉन ली: मुझे खुशी है कि यह उपयोगी था। मुझे बताने के लिए समय निकालने के लिए धन्यवाद: ' –

+7

यह मेरे लिए बहुत उपयोगी था। UITextView के साथ इसका उपयोग करने वाले अन्य लोगों के लिए दो बिंदु। पहले UITextView स्पष्ट रूप से आंतरिक रूप से स्वयं को संदर्भित करता है। डिलीगेट, इसलिए यह आपके मध्यस्थ को छोड़ देता है। फिक्स जब्बोट के "(आईडी) प्रतिनिधि को हटाने के लिए है ..." गेटटर लाइन (जिसका अर्थ है कि आपके बिचौलियों के प्रतिनिधि तरीकों को सीधे "असली" प्रतिनिधि को कॉल करना होगा)। दूसरा, यह जांचता है कि प्रतिनिधि उन्हें कॉल करने से पहले कुछ आंतरिक दिनचर्या लागू करता है, जो अनंत रिकर्सन की ओर जाता है। फिक्स जोड़ना है "अगर ([[मिडिलमैन सुपरक्लास] उदाहरण रीस्पॉन्डटोइलेक्टर: एसेलेक्टर]) वापस नहीं;" प्रतिक्रियाओं के शीर्ष पर चयनकर्ता: – mackworth

64

ई.जेम्स की पोस्ट ने अधिकांश विचारों के लिए एक उत्कृष्ट समाधान दिया। लेकिन यूआईटीएक्स्टफिल्ड और यूआईटीएक्स्टव्यू जैसे कीबोर्ड निर्भर विचारों के लिए, यह अक्सर अनंत लूप की स्थिति में परिणाम देता है। इससे छुटकारा पाने के लिए, मैंने इसे कुछ अतिरिक्त कोड के साथ तय किया है जो जांचता है कि चयनकर्ता विशिष्ट प्रोटोकॉल में है या नहीं।

WZProtocolInterceptor.h

#import <Foundation/Foundation.h> 

@interface WZProtocolInterceptor : NSObject 
@property (nonatomic, readonly, copy) NSArray * interceptedProtocols; 
@property (nonatomic, weak) id receiver; 
@property (nonatomic, weak) id middleMan; 

- (instancetype)initWithInterceptedProtocol:(Protocol *)interceptedProtocol; 
- (instancetype)initWithInterceptedProtocols:(Protocol *)firstInterceptedProtocol, ... NS_REQUIRES_NIL_TERMINATION; 
- (instancetype)initWithArrayOfInterceptedProtocols:(NSArray *)arrayOfInterceptedProtocols; 
@end 

WZProtocolInterceptor.m

#import <objc/runtime.h> 

#import "WZProtocolInterceptor.h" 

static inline BOOL selector_belongsToProtocol(SEL selector, Protocol * protocol); 

@implementation WZProtocolInterceptor 
- (id)forwardingTargetForSelector:(SEL)aSelector 
{ 
    if ([self.middleMan respondsToSelector:aSelector] && 
     [self isSelectorContainedInInterceptedProtocols:aSelector]) 
     return self.middleMan; 

    if ([self.receiver respondsToSelector:aSelector]) 
     return self.receiver; 

    return [super forwardingTargetForSelector:aSelector]; 
} 

- (BOOL)respondsToSelector:(SEL)aSelector 
{ 
    if ([self.middleMan respondsToSelector:aSelector] && 
     [self isSelectorContainedInInterceptedProtocols:aSelector]) 
     return YES; 

    if ([self.receiver respondsToSelector:aSelector]) 
     return YES; 

    return [super respondsToSelector:aSelector]; 
} 

- (instancetype)initWithInterceptedProtocol:(Protocol *)interceptedProtocol 
{ 
    self = [super init]; 
    if (self) { 
     _interceptedProtocols = @[interceptedProtocol]; 
    } 
    return self; 
} 

- (instancetype)initWithInterceptedProtocols:(Protocol *)firstInterceptedProtocol, ...; 
{ 
    self = [super init]; 
    if (self) { 
     NSMutableArray * mutableProtocols = [NSMutableArray array]; 
     Protocol * eachInterceptedProtocol; 
     va_list argumentList; 
     if (firstInterceptedProtocol) 
     { 
      [mutableProtocols addObject:firstInterceptedProtocol]; 
      va_start(argumentList, firstInterceptedProtocol); 
      while ((eachInterceptedProtocol = va_arg(argumentList, id))) { 
       [mutableProtocols addObject:eachInterceptedProtocol]; 
      } 
      va_end(argumentList); 
     } 
     _interceptedProtocols = [mutableProtocols copy]; 
    } 
    return self; 
} 

- (instancetype)initWithArrayOfInterceptedProtocols:(NSArray *)arrayOfInterceptedProtocols 
{ 
    self = [super init]; 
    if (self) { 
     _interceptedProtocols = [arrayOfInterceptedProtocols copy]; 
    } 
    return self; 
} 

- (void)dealloc 
{ 
    _interceptedProtocols = nil; 
} 

- (BOOL)isSelectorContainedInInterceptedProtocols:(SEL)aSelector 
{ 
    __block BOOL isSelectorContainedInInterceptedProtocols = NO; 
    [self.interceptedProtocols enumerateObjectsUsingBlock:^(Protocol * protocol, NSUInteger idx, BOOL *stop) { 
     isSelectorContainedInInterceptedProtocols = selector_belongsToProtocol(aSelector, protocol); 
     * stop = isSelectorContainedInInterceptedProtocols; 
    }]; 
    return isSelectorContainedInInterceptedProtocols; 
} 

@end 

BOOL selector_belongsToProtocol(SEL selector, Protocol * protocol) 
{ 
    // Reference: https://gist.github.com/numist/3838169 
    for (int optionbits = 0; optionbits < (1 << 2); optionbits++) { 
     BOOL required = optionbits & 1; 
     BOOL instance = !(optionbits & (1 << 1)); 

     struct objc_method_description hasMethod = protocol_getMethodDescription(protocol, selector, required, instance); 
     if (hasMethod.name || hasMethod.types) { 
      return YES; 
     } 
    } 

    return NO; 
} 

और यहाँ स्विफ्ट 2 संस्करण है:

// 
// NSProtocolInterpreter.swift 
// Nest 
// 
// Created by Manfred Lau on 11/28/14. 
// Copyright (c) 2014 WeZZard. All rights reserved. 
// 

import Foundation 

/** 
`NSProtocolInterceptor` is a proxy which intercepts messages to the middle man 
which originally intended to send to the receiver. 

- Discussion: `NSProtocolInterceptor` is a class cluster which dynamically 
subclasses itself to conform to the intercepted protocols at the runtime. 
*/ 
public final class NSProtocolInterceptor: NSObject { 
    /// Returns the intercepted protocols 
    public var interceptedProtocols: [Protocol] { return _interceptedProtocols } 
    private var _interceptedProtocols: [Protocol] = [] 

    /// The receiver receives messages 
    public weak var receiver: NSObjectProtocol? 

    /// The middle man intercepts messages 
    public weak var middleMan: NSObjectProtocol? 

    private func doesSelectorBelongToAnyInterceptedProtocol(
     aSelector: Selector) -> Bool 
    { 
     for aProtocol in _interceptedProtocols 
      where sel_belongsToProtocol(aSelector, aProtocol) 
     { 
      return true 
     } 
     return false 
    } 

    /// Returns the object to which unrecognized messages should first be 
    /// directed. 
    public override func forwardingTargetForSelector(aSelector: Selector) 
     -> AnyObject? 
    { 
     if middleMan?.respondsToSelector(aSelector) == true && 
      doesSelectorBelongToAnyInterceptedProtocol(aSelector) 
     { 
      return middleMan 
     } 

     if receiver?.respondsToSelector(aSelector) == true { 
      return receiver 
     } 

     return super.forwardingTargetForSelector(aSelector) 
    } 

    /// Returns a Boolean value that indicates whether the receiver implements 
    /// or inherits a method that can respond to a specified message. 
    public override func respondsToSelector(aSelector: Selector) -> Bool { 
     if middleMan?.respondsToSelector(aSelector) == true && 
      doesSelectorBelongToAnyInterceptedProtocol(aSelector) 
     { 
      return true 
     } 

     if receiver?.respondsToSelector(aSelector) == true { 
      return true 
     } 

     return super.respondsToSelector(aSelector) 
    } 

    /** 
    Create a protocol interceptor which intercepts a single Objecitve-C 
    protocol. 

    - Parameter  protocols: An Objective-C protocol, such as 
    UITableViewDelegate.self. 
    */ 
    public class func forProtocol(aProtocol: Protocol) 
     -> NSProtocolInterceptor 
    { 
     return forProtocols([aProtocol]) 
    } 

    /** 
    Create a protocol interceptor which intercepts a variable-length sort of 
    Objecitve-C protocols. 

    - Parameter  protocols: A variable length sort of Objective-C protocol, 
    such as UITableViewDelegate.self. 
    */ 
    public class func forProtocols(protocols: Protocol ...) 
     -> NSProtocolInterceptor 
    { 
     return forProtocols(protocols) 
    } 

    /** 
    Create a protocol interceptor which intercepts an array of Objecitve-C 
    protocols. 

    - Parameter  protocols: An array of Objective-C protocols, such as 
    [UITableViewDelegate.self]. 
    */ 
    public class func forProtocols(protocols: [Protocol]) 
     -> NSProtocolInterceptor 
    { 
     let protocolNames = protocols.map { NSStringFromProtocol($0) } 
     let sortedProtocolNames = protocolNames.sort() 
     let concatenatedName = sortedProtocolNames.joinWithSeparator(",") 

     let theConcreteClass = concreteClassWithProtocols(protocols, 
      concatenatedName: concatenatedName, 
      salt: nil) 

     let protocolInterceptor = theConcreteClass.init() 
      as! NSProtocolInterceptor 
     protocolInterceptor._interceptedProtocols = protocols 

     return protocolInterceptor 
    } 

    /** 
    Return a subclass of `NSProtocolInterceptor` which conforms to specified 
     protocols. 

    - Parameter  protocols:   An array of Objective-C protocols. The 
    subclass returned from this function will conform to these protocols. 

    - Parameter  concatenatedName: A string which came from concatenating 
    names of `protocols`. 

    - Parameter  salt:    A UInt number appended to the class name 
    which used for distinguishing the class name itself from the duplicated. 

    - Discussion: The return value type of this function can only be 
    `NSObject.Type`, because if you return with `NSProtocolInterceptor.Type`, 
    you can only init the returned class to be a `NSProtocolInterceptor` but not 
    its subclass. 
    */ 
    private class func concreteClassWithProtocols(protocols: [Protocol], 
     concatenatedName: String, 
     salt: UInt?) 
     -> NSObject.Type 
    { 
     let className: String = { 
      let basicClassName = "_" + 
       NSStringFromClass(NSProtocolInterceptor.self) + 
       "_" + concatenatedName 

      if let salt = salt { return basicClassName + "_\(salt)" } 
       else { return basicClassName } 
     }() 

     let nextSalt = salt.map {$0 + 1} 

     if let theClass = NSClassFromString(className) { 
      switch theClass { 
      case let anInterceptorClass as NSProtocolInterceptor.Type: 
       let isClassConformsToAllProtocols: Bool = { 
        // Check if the found class conforms to the protocols 
        for eachProtocol in protocols 
         where !class_conformsToProtocol(anInterceptorClass, 
          eachProtocol) 
        { 
         return false 
        } 
        return true 
        }() 

       if isClassConformsToAllProtocols { 
        return anInterceptorClass 
       } else { 
        return concreteClassWithProtocols(protocols, 
         concatenatedName: concatenatedName, 
         salt: nextSalt) 
       } 
      default: 
       return concreteClassWithProtocols(protocols, 
        concatenatedName: concatenatedName, 
        salt: nextSalt) 
      } 
     } else { 
      let subclass = objc_allocateClassPair(NSProtocolInterceptor.self, 
       className, 
       0) 
       as! NSObject.Type 

      for eachProtocol in protocols { 
       class_addProtocol(subclass, eachProtocol) 
      } 

      objc_registerClassPair(subclass) 

      return subclass 
     } 
    } 
} 

/** 
Returns true when the given selector belongs to the given protocol. 
*/ 
public func sel_belongsToProtocol(aSelector: Selector, 
    _ aProtocol: Protocol) -> Bool 
{ 
    for optionBits: UInt in 0..<(1 << 2) { 
     let isRequired = optionBits & 1 != 0 
     let isInstance = !(optionBits & (1 << 1) != 0) 

     let methodDescription = protocol_getMethodDescription(aProtocol, 
      aSelector, isRequired, isInstance) 

     if !objc_method_description_isEmpty(methodDescription) 
     { 
      return true 
     } 
    } 
    return false 
} 

public func objc_method_description_isEmpty(
    var methodDescription: objc_method_description) 
    -> Bool 
{ 
    let ptr = withUnsafePointer(&methodDescription) { UnsafePointer<Int8>($0) } 
    for offset in 0..<sizeof(objc_method_description) { 
     if ptr[offset] != 0 { 
      return false 
     } 
    } 
    return true 
} 
+2

मुझे वास्तव में इस कोड को पसंद है! मैं इसे 'UITextField' के लिए उपयोग करता हूं और यह बहुत अच्छा काम करता है। – Erik

+3

इस उत्तर को अब तक का सबसे अच्छा समाधान होने के लिए +100 बक्षीस देना था। देर से पोस्ट किए जाने के बाद इसे अधिक ध्यान देने की आवश्यकता थी। आपके द्वारा प्राप्त किए जा रहे अपवॉट्स का भारी प्रवाह इस प्रश्न का परिणाम [मेटा] से जुड़ा हुआ है (http://meta.stackoverflow.com/a/291533/2792531)। – nhgrif

+0

आपकी प्रशंसा के लिए धन्यवाद। ज्ञान साझा करने के लिए यह मेरी प्रेरणा है। – WeZZard

4

वास्तव में, यह मेरे लिए काम किया:

@implementation MySubclass { 
    id _actualDelegate; 
} 

// There is no need to set the value of _actualDelegate in an init* method 
- (void)setDelegate:(id)newDelegate { 
    [super setDelegate:nil]; 
    _actualDelegate = newDelegate; 
    [super setDelegate:(id)self]; 
} 

- (id)delegate { 
    return self; 
} 

- (id)forwardingTargetForSelector:(SEL)aSelector { 
    if ([_actualDelegate respondsToSelector:aSelector]) { return _actualDelegate; } 
    return [super forwardingTargetForSelector:aSelector]; 
} 

- (BOOL)respondsToSelector:(SEL)aSelector { 
    return [super respondsToSelector:aSelector] || [_actualDelegate respondsToSelector:aSelector]; 
} 
@end 

... उप-वर्ग को ई.जेम्स द्वारा दिए गए भयानक उत्तर में संदेश इंटरसेप्टर बनाने के लिए।

+0

मैंने इस जवाब को प्राथमिकता दी क्योंकि इसे किसी अन्य वर्ग की आवश्यकता नहीं है। निश्चित नहीं है कि क्यों प्रतिनिधि स्वयं को सेट करने से पहले शून्य पर सेट किया गया है। इसके अलावा वास्तविक डिलीगेट को इसे बनाए रखने से रोकने के लिए एक असाइन गुण के रूप में परिभाषित किया जाना चाहिए। – malhal

संबंधित मुद्दे