How to write iOS app purely in C - objective-c
I read here Learn C Before Objective-C?
Usually I then replace some Obj-C code with pure C code (after all you can mix them as much as you like, the content of an Obj-C method can be entirely, pure C code)
Is this true?
Is it possible to build an iPhone app purely in the C programming language?
Damn, it took me a while but I got it:
main.c:
#include <CoreFoundation/CoreFoundation.h>
#include <objc/runtime.h>
#include <objc/message.h>
// This is a hack. Because we are writing in C, we cannot out and include
// <UIKit/UIKit.h>, as that uses Objective-C constructs.
// however, neither can we give the full function declaration, like this:
// int UIApplicationMain (int argc, char *argv[], NSString *principalClassName, NSString *delegateClassName);
// So, we rely on the fact that for both the i386 & ARM architectures,
// the registers for parameters passed in remain the same whether or not
// you are using VA_ARGS. This is actually the basis of the objective-c
// runtime (objc_msgSend), so we are probably fine here, this would be
// the last thing I would expect to break.
extern int UIApplicationMain(int, ...);
// Entry point of the application. If you don't know what this is by now,
// then you probably shouldn't be reading the rest of this post.
int main(int argc, char *argv[])
{
// Create an #autoreleasepool, using the old-stye API.
// Note that while NSAutoreleasePool IS deprecated, it still exists
// in the APIs for a reason, and we leverage that here. In a perfect
// world we wouldn't have to worry about this, but, remember, this is C.
id autoreleasePool = objc_msgSend(objc_msgSend(objc_getClass("NSAutoreleasePool"), sel_registerName("alloc")), sel_registerName("init"));
// Notice the use of CFSTR here. We cannot use an objective-c string
// literal #"someStr", as that would be using objective-c, obviously.
UIApplicationMain(argc, argv, nil, CFSTR("AppDelegate"));
objc_msgSend(autoreleasePool, sel_registerName("drain"));
}
AppDelegate.c:
#import <objc/runtime.h>
#import <objc/message.h>
// This is equivalent to creating a #class with one public variable named 'window'.
struct AppDel
{
Class isa;
id window;
};
// This is a strong reference to the class of the AppDelegate
// (same as [AppDelegate class])
Class AppDelClass;
// this is the entry point of the application, same as -application:didFinishLaunchingWithOptions:
// note the fact that we use `void *` for the 'application' and 'options' fields, as we need no reference to them for this to work. A generic id would suffice here as well.
BOOL AppDel_didFinishLaunching(struct AppDel *self, SEL _cmd, void *application, void *options)
{
// we +alloc and -initWithFrame: our window here, so that we can have it show on screen (eventually).
// this entire method is the objc-runtime based version of the standard View-Based application's launch code, so nothing here really should surprise you.
// one thing important to note, though is that we use `sel_getUid()` instead of #selector().
// this is because #selector is an objc language construct, and the application would not have been created in C if I used #selector.
self->window = objc_msgSend(objc_getClass("UIWindow"), sel_getUid("alloc"));
self->window = objc_msgSend(self->window, sel_getUid("initWithFrame:"), (struct CGRect) { 0, 0, 320, 480 });
// here, we are creating our view controller, and our view. note the use of objc_getClass, because we cannot reference UIViewController directly in C.
id viewController = objc_msgSend(objc_msgSend(objc_getClass("UIViewController"), sel_getUid("alloc")), sel_getUid("init"));
// creating our custom view class, there really isn't too much
// to say here other than we are hard-coding the screen's bounds,
// because returning a struct from a `objc_msgSend()` (via
// [[UIScreen mainScreen] bounds]) requires a different function call
// and is finicky at best.
id view = objc_msgSend(objc_msgSend(objc_getClass("View"), sel_getUid("alloc")), sel_getUid("initWithFrame:"), (struct CGRect) { 0, 0, 320, 480 });
// here we simply add the view to the view controller, and add the viewController to the window.
objc_msgSend(objc_msgSend(viewController, sel_getUid("view")), sel_getUid("addSubview:"), view);
objc_msgSend(self->window, sel_getUid("setRootViewController:"), viewController);
// finally, we display the window on-screen.
objc_msgSend(self->window, sel_getUid("makeKeyAndVisible"));
return YES;
}
// note the use of the gcc attribute extension (constructor).
// Basically, this lets us run arbitrary code before program startup,
// for more information read here: http://stackoverflow.com/questions/2053029
__attribute__((constructor))
static void initAppDel()
{
// This is objc-runtime gibberish at best. We are creating a class with the
// name "AppDelegate" that is a subclass of "UIResponder". Note we do not need
// to register for the UIApplicationDelegate protocol, that really is simply for
// Xcode's autocomplete, we just need to implement the method and we are golden.
AppDelClass = objc_allocateClassPair(objc_getClass("UIResponder"), "AppDelegate", 0);
// Here, we tell the objc runtime that we have a variable named "window" of type 'id'
class_addIvar(AppDelClass, "window", sizeof(id), 0, "#");
// We tell the objc-runtime that we have an implementation for the method
// -application:didFinishLaunchingWithOptions:, and link that to our custom
// function defined above. Notice the final parameter. This tells the runtime
// the types of arguments received by the function.
class_addMethod(AppDelClass, sel_getUid("application:didFinishLaunchingWithOptions:"), (IMP) AppDel_didFinishLaunching, "i#:##");
// Finally we tell the runtime that we have finished describing the class and
// we can let the rest of the application use it.
objc_registerClassPair(AppDelClass);
}
View.c
#include <objc/runtime.h>
// This is a strong reference to the class of our custom view,
// In case we need it in the future.
Class ViewClass;
// This is a simple -drawRect implementation for our class. We could have
// used a UILabel or something of that sort instead, but I felt that this
// stuck with the C-based mentality of the application.
void View_drawRect(id self, SEL _cmd, struct CGRect rect)
{
// We are simply getting the graphics context of the current view,
// so we can draw to it
CGContextRef context = UIGraphicsGetCurrentContext();
// Then we set it's fill color to white so that we clear the background.
// Note the cast to (CGFloat []). Otherwise, this would give a warning
// saying "invalid cast from type 'int' to 'CGFloat *', or
// 'extra elements in initializer'. Also note the assumption of RGBA.
// If this wasn't a demo application, I would strongly recommend against this,
// but for the most part you can be pretty sure that this is a safe move
// in an iOS application.
CGContextSetFillColor(context, (CGFloat []){ 1, 1, 1, 1 });
// here, we simply add and draw the rect to the screen
CGContextAddRect(context, (struct CGRect) { 0, 0, 320, 480 });
CGContextFillPath(context);
// and we now set the drawing color to red, then add another rectangle
// and draw to the screen
CGContextSetFillColor(context, (CGFloat []) { 1, 0, 0, 1 });
CGContextAddRect(context, (struct CGRect) { 10, 10, 20, 20 });
CGContextFillPath(context);
}
// Once again we use the (constructor) attribute. generally speaking,
// having many of these is a very bad idea, but in a small application
// like this, it really shouldn't be that big of an issue.
__attribute__((constructor))
static void initView()
{
// Once again, just like the app delegate, we tell the runtime to
// create a new class, this time a subclass of 'UIView' and named 'View'.
ViewClass = objc_allocateClassPair(objc_getClass("UIView"), "View", 0);
// and again, we tell the runtime to add a function called -drawRect:
// to our custom view. Note that there is an error in the type-specification
// of this method, as I do not know the #encode sequence of 'CGRect' off
// of the top of my head. As a result, there is a chance that the rect
// parameter of the method may not get passed properly.
class_addMethod(ViewClass, sel_getUid("drawRect:"), (IMP) View_drawRect, "v#:");
// And again, we tell the runtime that this class is now valid to be used.
// At this point, the application should run and display the screenshot shown below.
objc_registerClassPair(ViewClass);
}
It's ugly, but it works.
If you would like to download this, you can get it from my dropbox here
You can get it from my GitHub repository here:
Objective-C is a superset of the C-language, so it is theoretically possible to write a program entirely in C, however, unless you are thoroughly versed in OpenGL ES, You'll need to do at least some objC (Even Rich's sample has a const NSString* in it), else you'll have to write the views yourself.
OK, the above is completely wrong. Let me say, I'm astounded Rich achieved this lofty goal, so I ported it over to the mac (source here). The files below have no headers, do not link to Cocoa, nor does the project have a nib:
AppDelegate.m
#include <objc/runtime.h>
#include <objc/message.h>
extern id NSApp;
struct AppDel
{
Class isa;
//Will be an NSWindow later, for now, it's id, because we cannot use pointers to ObjC classes
id window;
};
// This is a strong reference to the class of the AppDelegate
// (same as [AppDelegate class])
Class AppDelClass;
BOOL AppDel_didFinishLaunching(struct AppDel *self, SEL _cmd, id notification) {
//alloc NSWindow
self->window = objc_msgSend(objc_getClass("NSWindow"),
sel_getUid("alloc"));
//init NSWindow
//Adjust frame. Window would be about 50*50 px without this
//specify window type. We want a resizeable window that we can close.
//use retained backing because this thing is small anyhow
//return no because this is the main window, and should be shown immediately
self->window = objc_msgSend(self->window,
sel_getUid("initWithContentRect:styleMask:backing:defer:"),(NSRect){0,0,1024,460}, (NSTitledWindowMask|NSClosableWindowMask|NSResizableWindowMask|NSMiniaturizableWindowMask),NSBackingStoreRetained,NO);
//send alloc and init to our view class. Love the nested objc_msgSends!
id view = objc_msgSend(objc_msgSend(objc_getClass("View"), sel_getUid("alloc")), sel_getUid("initWithFrame:"), (struct CGRect) { 0, 0, 320, 480 });
// here we simply add the view to the window.
objc_msgSend(self->window, sel_getUid("setContentView:"), view);
objc_msgSend(self->window, sel_getUid("becomeFirstResponder"));
//makeKeyOrderFront: NSWindow to show in bottom left corner of the screen
objc_msgSend(self->window,
sel_getUid("makeKeyAndOrderFront:"),
self);
return YES;
}
static void initAppDel()
{
//Our appDelegate should be NSObject, but if you want to go the hard route, make this a class pair of NSApplication and try initing those awful delegate methods!
AppDelClass = objc_allocateClassPair((Class)
objc_getClass("NSObject"), "AppDelegate", 0);
//Change the implementation of applicationDidFinishLaunching: so we don't have to use ObjC when this is called by the system.
class_addMethod(AppDelClass,
sel_getUid("applicationDidFinishLaunching:"),
(IMP) AppDel_didFinishLaunching, "i#:#");
objc_registerClassPair(AppDelClass);
}
void init_app(void)
{
objc_msgSend(
objc_getClass("NSApplication"),
sel_getUid("sharedApplication"));
if (NSApp == NULL)
{
fprintf(stderr,"Failed to initialized NSApplication... terminating...\n");
return;
}
id appDelObj = objc_msgSend(
objc_getClass("AppDelegate"),
sel_getUid("alloc"));
appDelObj = objc_msgSend(appDelObj, sel_getUid("init"));
objc_msgSend(NSApp, sel_getUid("setDelegate:"), appDelObj);
objc_msgSend(NSApp, sel_getUid("run"));
}
//there doesn't need to be a main.m because of this little beauty here.
int main(int argc, char** argv)
{
//Initialize a valid app delegate object just like [NSApplication sharedApplication];
initAppDel();
//Initialize the run loop, just like [NSApp run]; this function NEVER returns until the app closes successfully.
init_app();
//We should close acceptably.
return EXIT_SUCCESS;
}
View.m
#include <objc/runtime.h>
#include <objc/message.h>
#include <ApplicationServices/ApplicationServices.h>
// This is a strong reference to the class of our custom view,
// In case we need it in the future.
Class ViewClass;
// This is a simple -drawRect implementation for our class. We could have
// used a UILabel or something of that sort instead, but I felt that this
// stuck with the C-based mentality of the application.
void View_drawRect(id self, SEL _cmd, CGRect rect)
{
//make a red NSColor object with its convenience method
id red = objc_msgSend(objc_getClass("NSColor"), sel_getUid("redColor"));
// fill target rect with red, because this is it!
NSRect rect1 = NSMakeRect ( 21,21,210,210 );
objc_msgSend(red, sel_getUid("set"));
NSRectFill ( rect1 );
}
// Once again we use the (constructor) attribute. generally speaking,
// having many of these is a very bad idea, but in a small application
// like this, it really shouldn't be that big of an issue.
__attribute__((constructor))
static void initView()
{
// Once again, just like the app delegate, we tell the runtime to
// create a new class, this time a subclass of 'UIView' and named 'View'.
ViewClass = objc_allocateClassPair((Class) objc_getClass("NSView"), "View", 0);
// and again, we tell the runtime to add a function called -drawRect:
// to our custom view. Note that there is an error in the type-specification
// of this method, as I do not know the #encode sequence of 'CGRect' off
// of the top of my head. As a result, there is a chance that the rect
// parameter of the method may not get passed properly.
class_addMethod(ViewClass, sel_getUid("drawRect:"), (IMP) View_drawRect, "v#:");
// And again, we tell the runtime that this class is now valid to be used.
// At this point, the application should run and display the screenshot shown below.
objc_registerClassPair(ViewClass);
}
prefix.pch
//
// Prefix header for all source files of the 'CBasedMacApp' target in the 'CBasedMacApp' project
//
#ifdef __OBJC__
#import <Foundation/Foundation.h>
#import <AppKit/AppKit.h>
#endif
I read here Learn C Before Objective-C?
Usually I then replace some Obj-C code with pure C code (after all you can mix them as much as you like, the content of an Obj-C method can be entirely, pure C code)
Is this true?
Could I build an iPhone app purely in the C programming language?
The quoted passage is true, but the answer to your question is no.
To illustrate what answerer Mecki on that other question was talking about:
- (void) drawRect:(CGRect)dirtyRect { //Objective-C
CGContextRef context = UIGraphicsGetCurrentContext(); //C
CGContextSetRGBFillColor(context, 1.0, 0.0, 0.0, 1.0); //C
CGContextFillRect(context, dirtyRect); //C
} //Objective-C (balances above “- (void) drawRect:…” line)
There is nothing but pure C code within this method, but the method itself is Objective-C code, as is the class that contains this method.
So it is possible to do what Mecki said, but you can't (practically—as Richard J. Ross III showed, it's technically possible but quite a lot of typing) write a whole Cocoa Touch program in pure C.
Actually, some of the code posted here, while written in C, is still calling objective-C code :). I don't know if that actually fits the scenario from the original poster when he asked
Is it possible to build an iPhone app purely in the C programming
language?
but I would agree with the people saying that, generally speaking and for an app with a GUI, you would need to write your GUI in OpenGL (which is C).
I think that is what most games do, right? Although I'm not sure if there's access to the iPhone's I/O (the touchscreen for example) in C.
Last but not least, the guys that wrote the code above rock! :)
Related
MacOS finding classes at launchtime
Is there a way to determine which classes are present at launch time? I have a Swift OS-X app that offers an API to add functionality. Not dynamically though, everything is fixed at launch time. But is it possible to find out which classes are present at launch, and then from there to call a certain (inherited) static operation on these classes? Absent this possibility I will have to create an initialisation routine that must be updated every time a new subclass is added. I would like to avoid this. Clarification: I have a protocol call it MyProtocol. At launch of the App I would like to call an operation from MyProtocol on all classes that implement MyProtocol. After examining the runtime manual I have gotten as far as: #import <Foundation/Foundation.h> #include "Test.h" #import <objc/objc-class.h> void activateLaunchActions() { // Get a list of all classes int numClasses = 0, newNumClasses = objc_getClassList(NULL, 0); Class *classes = NULL; while (numClasses < newNumClasses) { numClasses = newNumClasses; Class newClasses[numClasses]; classes = newClasses; newNumClasses = objc_getClassList(classes, numClasses); } // Get the protocol they have to confirm to Protocol *prot = objc_getProtocol("MyProtocol"); // Get the selector to be called SEL sel = sel_registerName("launchAction"); // Create the launchAction caller from objc_msgSend typedef void (*send_type)(Class, SEL); send_type callLauchAction = (send_type)objc_msgSend; // Call the registration for all classes that confirm to the protocol for (int i=0; i<numClasses; i++) { if (class_conformsToProtocol(classes[i], prot)) { callLauchAction(classes[i], sel); } } } The "prot" is always NULL. Hence the test later fails and the launchAction is never called. I have read that protocol objects are not always present if they are not used by a class, however this protocol is used by several classes. Any suggestions as how to fix this would be welcome.
Does the self keyword have any special meaning for a C struct?
Trying to make sense of MTAudioProcessingTap sample code from WWDC 2012. This structure is used by the Tap to pass data around. typedef struct AVAudioTapProcessorContext { ... void *self; //Note use of word self } AVAudioTapProcessorContext; The word self here is used apparently as a standard variable name even though Xcode highlights it in pink. Is this just a code parsing oversight in Xcode? Later, the Tap management class passes a reference to itself to this variable. //Setting up Tap callbacks: callbacks.clientInfo = (__bridge void *)self, //And then storing it in the above struct when it's passed by the callback: static void tap_InitCallback(MTAudioProcessingTapRef tap, void *clientInfo, void **tapStorageOut) { AVAudioTapProcessorContext *context = calloc(1, sizeof(AVAudioTapProcessorContext)); ... context->self = clientInfo; *tapStorageOut = context; } And the confusion starts again in a later function (another Tap callback). Once again, the word self is used to refer to the Tap management class reference. Note that this occurs in a function defined after #end of the Tap management class implementation. AudioFFTTapProcessor *self = ((__bridge AudioFFTTapProcessor *)context->self);
self has a special meaning only in the context of an Objective-C method. In all other contexts, it's just an ordinary identifier. You can think of Objective-C methods as being rewritten to take a hidden parameter with the name self (and a second hidden parameter named _cmd that's not relevant for this discussion), e.g.: #implementation MyClass - (void)doSomething:(int)param1 withThis:(NSString *)param2 { // 'self' refers to the MyClass instance in this function ... } #end could be imagined as if it were rewritten like this: void MyClass_doSomething(id self, SEL _cmd, int param1, NSString *param2) { // 'self' refers to the MyClass instance in this function ... } Outside of Objective-C methods, you can use self as any other kind of identifier, whether it be a local variable, a struct member, a global function, or type name, etc., though the latter uses should be highly discouraged due to high potential for confusion. So yes, XCode highlighting the self instance in pink in the struct definition is just a parsing anomaly.
self is not a keyword in C. It's a keyword for Objective-C, which is probably why it's getting highlighted.
self does not mean anything special as a C struct field name. Objective-C uses it as the equivalent of most other OO languages' this, which is why Xcode highlights it specially.
Objective-C - iVar Scoped Method Variables?
I was messing around in Objective-C earlier, and I ran into a quite common situation: I had a class, which was not a singleton, that needed a variable shared between method calls, like static, but each instance needed it's own variable. However, this variable only needed to be used in one particular method, we'll call it -foo. What I'd love to do, is have a macro, let's call it ivar, which lets me do the following: #implementation MyClass -(foo) { ivar int someVal = 10; // default value, ivar scoped variable. } -(bar) { someVal = 5; // error, outside of `foo`'s scope. } #end How the variable is defined does not matter to me (either a macro like OBJC_IVAR(Type, Name, Default) or ivar someType someName = value), as long as it meets the following requirements: Has thread safety Can have variable of same name (but different value) in another method Type-less (doesn't matter what type the variable is) Default Value support Variable can be declared in one line (I shouldn't have to write 15 lines of code just to put a variable in my code) I am currently working on an Objective-C++ implementation myself, I was just wondering if anyone else had any thoughts (or existing tools) on how to do this. Obviously, this doesn't have to be done with a true iVar. More likely, this should be done with associated objects at run-time, which also manages deallocation for us.
After a lot of time spent, I believe I have a fully working solution in Objective-C++. Some of the features: The variables are unique. As long as they have a different scope, their values are independent Each instance has it's own values Thread safety (accomplished by associated objects) Simple variable declaration: Macro overloading: only specify the information that you need Possible ways to define an OBJC_IVAR: OBJC_IVAR(); // creates a warning, does nothing OBJC_IVAR(Name); // creates an ivar named 'Name' of type 'id' OBJC_IVAR(Type, Name); // creates an ivar named 'Name' of type 'Type' OBJC_IVAR(Type, Name, Default); // creates an ivar named 'Name', of type 'Type', and a default value of 'Default' (which is only executed once); Full Type Support with C++ templates (__weak, __strong, __autoreleasing, volatile, etc. are all supported) Subclasses do not share variables with their superclasses (so no chance for conflicts, variables really are limited to their scope). Can be used in singletons without issue Is fast, takes ~15-30 CPU cycles to look up a variable, and once it's looked up, takes as long as any other variable to set it. Most of the hard work is done by the pre-processor, which allows for faster code Just drag-and-drop into an existing Xcode project, doesn't rely on a custom processor Some minor cons to the implementation: Objects must have an ownership specifier (limitation with C++ references: Reference to non-const type 'id' with no explicit ownership). Is easily fixed by adding __strong, __weak, or __autoreleasing to the type of the variable Implementation is hard to read. Because it relies so much on C++ templates and Objective-C working together in harmony, it's difficult to just change 'one thing' and hope for it to work. I have added extensive comments to the implementation, so hopefully that frees some of the burden. Method swizzling can confuse this majorly. Not the largest of issues, but if you start playing around with method swizzling, don't be surprised if you get unexpected results. Cannot be used inside a C++ object. Unfortunately, C++ doesn't support runtime attributes, like objective-c does, so we cannot rely upon our variables being cleaned up eventually. For this reason, you cannot use OBJC_IVAR while inside a C++ object. I would be interested in seeing an implementation for that, though. #line can mess this up drastically, so don't use it. Version History 1.0: Initial Release 1.1: Updated OBJC_IVAR_NAME to rely only on the preprocessor. As a result, we cannot use __func__. So, without further ado, here is the code: OBJC_IVAR.hpp // // OBJC_IVAR.h // TestProj // // Created by Richard Ross on 8/17/12. // Copyright (c) 2012 Ultimate Computer Services, Inc. All rights reserved. // #ifndef OBJC_IVAR_HPP #define OBJC_IVAR_HPP #import <Foundation/Foundation.h> #import <objc/runtime.h> #import "NSValue+CppObject.h" // Argument counting algorithm. Not too complex #define __NARG(_1, _2, _3, _4, _5, VAL, ...) VAL #define NARG(...) __NARG(__VA_ARGS__, 5, 4, 3, 2, 1, 0) // Different implementations based on number of parameters passed in #define __OBJC_IVAR(N, ...) _OBJC_IVAR_ ## N (__VA_ARGS__) #define _OBJC_IVAR(N, ...) __OBJC_IVAR(N, __VA_ARGS__) // Usage: OBJC_IVAR(Type (optional), Name (required), Default (optional)) #define OBJC_IVAR(...) _OBJC_IVAR(NARG(__VA_ARGS__), __VA_ARGS__) // create a unique name. we use '__COUNTER__' here to support scoping on the same line, for compressed source code #define __OBJC_IVAR_STRINGIFY_NAME(file, line, name, counter) #file ":" #line " " #name ":" #counter #define _OBJC_IVAR_NAME(file, line, name, counter) __OBJC_IVAR_STRINGIFY_NAME(file, line, name, counter) #define OBJC_IVAR_NAME(name) _OBJC_IVAR_NAME(__FILE__, __LINE__, name, __COUNTER__) // old style creation. advantage: uses __func__ to determine calling function // #define OBJC_IVAR_NAME(Name) [NSString stringWithFormat:#"%s:%i %s:%s:%i", __FILE__, __LINE__, __func__, #Name, __COUNTER__] // implemenations for each of the overloads #define _OBJC_IVAR_0(...) _Pragma("message \"Cannot call OBJC_IVAR with 0 params!\"") #define _OBJC_IVAR_1(Name) _OBJC_IVAR_2(__strong id, Name) // first major implemenation. because we do no assignment here, we don't have to check for is_set #define _OBJC_IVAR_2(Type, Name) Type& Name = (_OBJC_IVAR::IMPL<Type>(self, OBJC_IVAR_NAME(Name))) // this is where things get fun. we have 'OBJC_IVAR_CUR_NAME', instead of calling OBJC_IVAR_NAME // multiple times, because we must ensure that COUNTER does not change during the course of the macro // this is the 'inner bowels' of C, and it's quite hacky. Returns a reference to an associated object // which is wrapped in a NSValue. Note that we only evaluate 'default' once throught the course of the // application's cycle, so you can feel free to put intensive loading code there. static NSString *_OBJC_IVAR_CUR_NAME; #define _OBJC_IVAR_3(Type, Name, Default) Type& Name = (_OBJC_IVAR::IS_SET(self, (_OBJC_IVAR_CUR_NAME = OBJC_IVAR_NAME(Name))) ? _OBJC_IVAR::IMPL<Type>(self, _OBJC_IVAR_CUR_NAME) : _OBJC_IVAR::IMPL<Type>(self, _OBJC_IVAR_CUR_NAME, Default)) // namespace to wrap al lof our functions namespace _OBJC_IVAR { // internal dictionary of all associated object names, so that we don't run // into memory management issues. we use a set here, because we should never // have duplicate associated object names. static NSMutableSet *_names = [NSMutableSet set]; // wraps a value and a reference to a value. used over std::reference_wrapper, // as that doesn't actually copy in the value passed. That is required for what // we are doing, as we cannot be assigning to constants. template<typename T> class Wrapper { private: // private value wrapped by this object. T _value; // private reference wrapped by this object. should always point to _value. T& _ref; public: // default constructor. assumes 'T' has a valid 0-argument constructor Wrapper() : _value(), _ref(_value) { } // argument constructor. makes sure that value is initialized properly Wrapper(T val) : _value(val), _ref(_value) { } // returns the reference wrapped by this object operator T& () { return _ref; } T& get() { return _ref; } }; // interns a name. because objc_getAssociatedObject works only by comparing // pointers (and +stringWithFormat: isn't guaranteed to return the same pointer), // we have to make sure that we maintain a list of all valid associated object // names. these are NOT linked to specific objects, which allows us to reuse some // memory inline NSString *name_intern(NSString *name) { // intern the value. first check if the object has been interned already, // and if it is, return that interned value if (id tmpName = [_names member:name]) { name = tmpName; } // if we haven't interned this value before, then add it to the list and return it. else { [_names addObject:name]; } return name; } // check and see if the requested iVar has been set yet. used for default value setting BOOL IS_SET(id target, NSString *name) { // first intern the name name = name_intern(name); // check if the object has this property. objc_getAssociatedObject will ALWAYS // return NULL if the object doesn't exist. Note the bridged cast. This is because // objc_getAssociatedObject doesn't care what you throw into the second parameter, // as long as it is a pointer. That gives us the flexibility at a later date, to, // for example, just pass a pointer to a single byte, and pull out the value that // way. However, we pass in a NSString pointer, because it makes it easy for us to // use and to re-use later. id val = objc_getAssociatedObject(target, (__bridge const void *) name); return val != nil; } // the actual implementation for setting the iVar. luckily this code isn't too hacky, // but it is a bit confusing. template<typename T> Wrapper<T>& IMPL(id target, NSString *name) { // first intern the name name = name_intern(name); // define a reference. we use pointers & new here, because C++ memory managment is // weird at best. Most of the time, you should be using RAII, but when dealing with // templates & objective-c interpolation, it is almost required that you use pointers // with new. Wrapper<T> *reference = nullptr; // check and see if the object already contains this property, if so, return that value NSValue *result = objc_getAssociatedObject(target, (__bridge const void *) name); if (result == nil) { // at this point, we need to create a new iVar, with the default constructor for the type. // for objective-c objects this is 'nil', for integers and floating point values this is 0, // for C++ structs and classes, this calls the default constructor. If one doesn't exist, // you WILL get a compile error. reference = new Wrapper<T>(); // we now set up the object that will hold this wrapper. This is an extension on NSValue // which allows us to store a generic pointer (in this case a C++ object), and run desired // code on -dealloc (which will be called at the time the parent object is destroyed), in // this case, free the memory used by our wrapper. result = [NSValue valueWithCppObject:reference onDealloc:^(void *) { delete reference; }]; // finally, set the associated object to the target, and now we are good to go. // We use OBJC_ASSOCIATION_RETAIN, so that our NSValue is properly freed when done. objc_setAssociatedObject(target, (__bridge const void *) name, result, OBJC_ASSOCIATION_RETAIN); } // from result, we cast it's -cppObjectValue to a Wrapper, to pull out the value. reference = static_cast<Wrapper<T> *>([result cppObjectValue]); // finally, return the pointer as a reference, not a pointer return *reference; } // this is pretty much the same as the other IMPL, but it has specific code for default values. // I will ignore everything that is the same about the two functions, and only focus on the // differences, which are few, but mandatory. template<typename T> Wrapper<T>& IMPL(id target, NSString *name, const T& defVal) { name = name_intern(name); Wrapper<T> *reference = nullptr; // asign to be the default constructor for 'T' NSValue *result = objc_getAssociatedObject(target, (__bridge const void *) name); if (result == nil) { // this is the only difference. Instead of constructing with the default constructor, // simply pass in our new default value as a copy. reference = new Wrapper<T>(defVal); result = [NSValue valueWithCppObject:reference onDealloc:^(void *) { delete reference; }]; objc_setAssociatedObject(target, (__bridge const void *) name, result, OBJC_ASSOCIATION_RETAIN); } reference = static_cast<Wrapper<T> *>([result cppObjectValue]); return *reference; } } #endif // OBJC_IVAR_HPP NSValue+CppObject.h // // NSValue+CppObject.h // TestProj // // Created by Richard Ross on 8/17/12. // Copyright (c) 2012 Ultimate Computer Services, Inc. All rights reserved. // #import <Foundation/Foundation.h> // Extension on NSValue to add C++ object support. Because of the difficulty // involved in templates, I took the easy way out and simply passed in a block // of code to be run at dealloc. #interface NSValue (CppObject) // create a new NSValue instance that holds ptr, and calls 'deallocBlock' on destruction. +(id) valueWithCppObject:(void *) ptr onDealloc:(void (^)(void *)) deallocBlock; -(id) initWithCppObject:(void *) ptr onDealloc:(void (^)(void *)) deallocBlock; // get the held pointer of this object. I called it -cppObjectValue, so // there was no confusion with -pointerValue. -(void *) cppObjectValue; #end NSValue+CppObject.m // // NSValue+CppObject.m // TestProj // // Created by Richard Ross on 8/17/12. // Copyright (c) 2012 Ultimate Computer Services, Inc. All rights reserved. // #import "NSValue+CppObject.h" // the concrete NSValue subclass for supporting C++ objects. Pretty straight-forward interface. #interface ConcreteCppObject : NSValue { // the underlying object that is being pointed to void *_object; // the block that is called on -dealloc void (^_deallocBlock)(void *); } #end #implementation ConcreteCppObject // object initialization +(id) valueWithCppObject:(void *)ptr onDealloc:(void (^)(void *))deallocBlock { return [[self alloc] initWithCppObject:ptr onDealloc:deallocBlock]; } -(id) initWithCppObject:(void *)ptr onDealloc:(void (^)(void *))deallocBlock { if (self = [super init]) { _object = ptr; _deallocBlock = deallocBlock; } return self; } // required methods for subclassing NSValue -(const char *) objCType { return #encode(void *); } -(void) getValue:(void *)value { *((void **) value) = _object; } // comparison -(BOOL) isEqual:(id)compare { if (![compare isKindOfClass:[self class]]) return NO; return [compare cppObjectValue] == [self cppObjectValue]; } // cleanup -(void) dealloc { // this should manage cleanup for us _deallocBlock(_object); } // value access -(void *) cppObjectValue { return _object; } #end // NSValue additions for creating the concrete instances #implementation NSValue (CppObject) // object initialization +(id) valueWithCppObject:(void *)ptr onDealloc:(void (^)(void *))deallocBlock { return [[ConcreteCppObject alloc] initWithCppObject:ptr onDealloc:deallocBlock]; } -(id) initWithCppObject:(void *)ptr onDealloc:(void (^)(void *))deallocBlock { return [[self class] valueWithCppObject:ptr onDealloc:deallocBlock]; } // unless the NSValue IS a ConcreteCppObject, then we shouldn't do anything here -(void *) cppObjectValue { [self doesNotRecognizeSelector:_cmd]; return nil; } #end Example Usage: #import "OBJC_IVAR.hpp" #interface SomeObject : NSObject -(void) doSomething; #end #implementation SomeObject -(void) doSomething { OBJC_IVAR(__strong id, test, #"Hello World!"); OBJC_IVAR(int, test2, 15); NSLog(#"%#", test); NSLog(#"%i", test2 += 7); // new scope { OBJC_IVAR(int, test, 100); NSLog(#"%i", ++test); } [self somethingElse]; } -(void) somethingElse { OBJC_IVAR(int, newVar, 7); NSLog(#"%i", newVar++); } #end int main() { SomeObject *obj = [SomeObject new]; [obj doSomething]; [obj doSomething]; [obj doSomething]; }
I had a class, which was not a singleton, that needed a variable shared between method calls, like static, but each instance needed it's own variable. In that case, the variable is part of the object's state, and it's therefore most appropriate to use an instance variable (or a property). This is exactly what ivars are for, whether they're used in a dozen methods or just one. I am currently working on an Objective-C++ implementation myself, I was just wondering if anyone else had any thoughts (or existing tools) on how to do this. My advice is to not do it at all. If your goal is to avoid clutter, don't go needlessly trying to add a new storage class to the language. However, if you're determined to pursue this line, I'd look at using blocks instead of associated objects. Blocks get their own copies of variables that are scoped to the lifetime of the block. For example, you can do this: - (void)func { __block int i = 0; void (^foo)() = ^{ i++; NSLog(#"i = %d", i); }; foo(); foo(); foo(); } and the output you get is: i = 1 i = 2 i = 3 Perhaps you can find a clever way to wrap that up in a macro, but it looks to me like a lot of trouble just to avoid declaring an instance variable.
How to add a category to a "hidden" class
Is there a way to add a category to a class whose header file you can't access? For testing purposes, I want to add a category to UITableViewCellDeleteConfirmationControl, but the class is (as far as I can tell) part of a private framework. How can I do that? Elaboration (per mihirios's request): I am trying to extend the Frank testing framework to simulate tapping the confirmation button (the big red "Delete" button) that appears when you try to delete a UITableViewCell. Frank adds a tap method to UIControl. For some reason, Frank's usual way of tapping a control does not work for the UITableViewCellDeleteConfirmationControl class (which subclasses UIControl). I've create a workaround. I added a category to UITableViewCell, with the following method. - (BOOL)confirmDeletion { if (![self showingDeleteConfirmation]) { return NO; } UITableView *tableView = (UITableView *)[self superview]; id <UITableViewDataSource> dataSource = [tableView dataSource]; NSIndexPath *indexPath = [tableView indexPathForCell:self]; [dataSource tableView:tableView commitEditingStyle:UITableViewCellEditingStyleDelete forRowAtIndexPath:indexPath]; return YES; } This finds the table's data source and invokes its tableView:commitEditingStyle:forRowAtIndexPath: method, which (according to the documentation for UITableView) is what the system does when the user taps the confirmation button. This works, but I would prefer to make UITableViewCellDeleteConfirmationControl appear to be a tappable button by adding a tap method to it, overriding Frank's default one. The tap method would find the cell that contains the confirmation button, then invoke [cell confirmDeletion]. When I try to declare a category for UITableViewCellDeleteConfirmationControl, the compiler complains that it "can't resolve interface 'UITableViewCellDeleteConfirmationControl'." When I try to use the header file that someone generated using class-dump, the linker complains that it can't find the symbol _OBJC_CLASS_$_UITableViewCellDeleteConfirmationControl.
For testing purposes, you can always get the class object using NSClassFromString and then use the class_replaceMethod runtime method to do whatever you need. See the Objective-C Runtime Reference for details.
As far as i know you can not use a Category, but you could add the methods manually during runtime. A Possible way to do this is, to create a new class, implement the methods you want to, and send this methods to UITableViewCellDeleteConfirmationControl using the appropriate objc-runtime functions. There are some things to take care of, like storing the original functions for later use in case of overloading, also in your 'category'-class you have to pay attention when you want to call super, as this will not work, you have to use objc-runtime function objc_msgSendSuper instead. As Long as you don't need to call super this will do fine: #import <objc/runtime.h> #import <objc/message.h> void implementInstanceMethods(Class src, Class dest) { unsigned int count; Method *methods = class_copyMethodList(src, &count); for (int i = 0; i < count; ++i) { IMP imp = method_getImplementation(methods[i]); SEL selector = method_getName(methods[i]); NSString *selectorName = NSStringFromSelector(selector); const char *types = method_getTypeEncoding(methods[i]); class_replaceMethod(dest, selector, imp, types); } free(methods); } a good point to call the method is in main.m, for example: #autoreleasepool { implementInstanceMethods([MyCategory class], NSClassFromString(#"UITableViewCellDeleteConfirmationControl")); return UIApplicationMain(argc, argv, nil, NSStringFromClass([YourAppDelegate class])); } But i don't know why you not just move the confirmation handling in the controller-class.
As long as the compiler can (eventually) link to the class in question you can create a category for it. The more important question will be how to design the category since it seems you do not have access to the source for the original class.
Objective-C, how can i hook up a method in another class
Objective-C keeps all its methods in a huge hashtable - so shouldn't it possible to patch this table and replace an existing method with my own patched method (which then calls the original)? I need a way to hook up the NSWindow KeyUp method in a window which i can't subclass cause it's already created. I need some code or at least some keywords i can use for further searching.
You should NOT swizzle methods for this. This is deprecated behavior. This will affect ALL windows in your app not just the one you wanted to change. However, what you should do instead is to subclass NSWindow already and then change the class of that window at runtime. This can be done using this runtime function: Class object_setClass(id object, Class cls) Reference is here: http://developer.apple.com/mac/library/documentation/Cocoa/Reference/ObjCRuntimeRef/Reference/reference.html#//apple_ref/doc/uid/TP40001418-CH1g-SW12 Your code should then look like this: object_setClass(theWindow, [MyWindowSubclass class]); On problem you might experience is that window already being a subclass of NSWindow. If that's the case there are more complicated ways to achieve this. You can construct a class dynamically at runtime. Here's some more code. Given that window is the target window: Class newWindowClass = objc_allocateClassPair([window class], "MyHackyWindowSubclass", 0); Method upMethod = class_getInstanceMethod(newWindowClass, #selector(keyUp:)); method_setImplementation(upMethod, new_NSWindow_keyUp_); object_setClass(window, newWindowClass); I'm not totally sure this does not change the implementation of the superclass. The documentation is a bit unspecific about it. However, you should still try it. If it does not work, replace the second and third line by this one: class_replaceMethod(newWindowClass, #selector(keyUp:), new_NSWindow_keyUp_, "v#:#"); In any case you need to define the new Method implementation. It could look like that (partially by KennyTM): void new_NSWindow_keyUp_(NSWindow* self, SEL _cmd, NSEvent* evt) { [super keyUp: evt]; ... // do your changes }
Of course it is possible. In fact, you don't even need to look into the hash table — there's standard API for this. For example: typedef void (*NSWindow_keyUp__IMP)(NSWindow* self, SEL _cmd, NSEvent* evt); static NSWindow_keyUp__IMP original_NSWindow_keyUp_; void replaced_NSWindow_keyUp_(NSWindow* self, SEL _cmd, NSEvent* evt) { NSLog(#"Entering keyUp:. self = %#, event = %#", self, evt); original_NSWindow_keyUp_(self, _cmd, evt); NSLog(#"Leaving keyUp:. self = %#, event = %#", self, evt); } ... Method m = class_getInstanceMethod([NSWindow class], #selector(keyUp:)); original_NSWindow_keyUp_ = method_setImplementation(m, replaced_NSWindow_keyUp_);