| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189 |
- import { describe, it, expect } from 'vitest';
- import {
- objcSelectorForSwiftMethod,
- objcSelectorForSwiftInit,
- objcAccessorsForSwiftProperty,
- swiftBaseNamesForObjcSelector,
- detectExplicitObjcName,
- isObjcExposed,
- } from '../src/resolution/swift-objc-bridge';
- describe('Swift → ObjC selector bridging (auto-name rules)', () => {
- describe('objcSelectorForSwiftMethod', () => {
- it('no parameters → bare base name', () => {
- expect(objcSelectorForSwiftMethod('play', [])).toBe('play');
- });
- it('single _ param → base + ":"', () => {
- expect(objcSelectorForSwiftMethod('play', ['_'])).toBe('play:');
- expect(objcSelectorForSwiftMethod('play', [null])).toBe('play:');
- });
- it('single labeled param → "baseWithLabel:"', () => {
- expect(objcSelectorForSwiftMethod('play', ['song'])).toBe('playWithSong:');
- });
- it('multi-param with leading _ → "base:label2:..."', () => {
- expect(objcSelectorForSwiftMethod('play', ['_', 'by'])).toBe('play:by:');
- expect(
- objcSelectorForSwiftMethod('tableView', ['_', 'didSelectRowAtIndexPath'])
- ).toBe('tableView:didSelectRowAtIndexPath:');
- });
- it('multi-param with leading explicit label → "baseWithFirst:rest:"', () => {
- expect(objcSelectorForSwiftMethod('play', ['song', 'by'])).toBe(
- 'playWithSong:by:'
- );
- });
- it('@objc(custom:) overrides the rule literally', () => {
- expect(
- objcSelectorForSwiftMethod('whateverName', ['ignored'], 'custom:')
- ).toBe('custom:');
- });
- it('returns null on empty base name', () => {
- expect(objcSelectorForSwiftMethod('', [])).toBeNull();
- });
- });
- describe('objcSelectorForSwiftInit', () => {
- it('init() → "init"', () => {
- expect(objcSelectorForSwiftInit([], [])).toBe('init');
- });
- it('init(name:) → "initWithName:"', () => {
- expect(objcSelectorForSwiftInit(['name'], ['name'])).toBe('initWithName:');
- });
- it('init(name:, age:) → "initWithName:age:"', () => {
- expect(objcSelectorForSwiftInit(['name', 'age'], ['name', 'age'])).toBe(
- 'initWithName:age:'
- );
- });
- it('init(_ name:) uses internal name → "initWithName:"', () => {
- expect(objcSelectorForSwiftInit(['_'], ['name'])).toBe('initWithName:');
- });
- it('@objc(custom) override on init', () => {
- expect(objcSelectorForSwiftInit(['name'], ['name'], 'custom:')).toBe(
- 'custom:'
- );
- });
- });
- describe('objcAccessorsForSwiftProperty', () => {
- it('getter = name, setter = setName:', () => {
- expect(objcAccessorsForSwiftProperty('name')).toEqual({
- getter: 'name',
- setter: 'setName:',
- });
- });
- it('camelCase → set capitalizes first', () => {
- expect(objcAccessorsForSwiftProperty('isReady')).toEqual({
- getter: 'isReady',
- setter: 'setIsReady:',
- });
- });
- it('explicit @objc(custom) overrides getter name', () => {
- expect(objcAccessorsForSwiftProperty('name', 'displayName')).toEqual({
- getter: 'displayName',
- setter: 'setDisplayName:',
- });
- });
- });
- });
- describe('ObjC selector → Swift base name candidates (reverse map)', () => {
- it('bare no-colon selector → itself', () => {
- expect(swiftBaseNamesForObjcSelector('play')).toEqual(['play']);
- });
- it('"play:" → ["play"]', () => {
- expect(swiftBaseNamesForObjcSelector('play:')).toEqual(['play']);
- });
- it('"playWithSong:" → ["playWithSong", "play"]', () => {
- expect(swiftBaseNamesForObjcSelector('playWithSong:').sort()).toEqual(
- ['play', 'playWithSong'].sort()
- );
- });
- it('Cocoa-style "objectForKey:" → includes "object"', () => {
- expect(swiftBaseNamesForObjcSelector('objectForKey:')).toContain('object');
- });
- it('Cocoa-style "stringWithFormat:" → includes "string"', () => {
- expect(swiftBaseNamesForObjcSelector('stringWithFormat:')).toContain('string');
- });
- it('Cocoa-style "imageNamed:inBundle:" → first keyword has no preposition, falls through', () => {
- // First keyword is `imageNamed` — no With/For/By in it, so candidates is
- // just the raw keyword. (`Named` is not in our preposition list — keep
- // it that way, otherwise we over-match on perfectly normal verbs.)
- expect(swiftBaseNamesForObjcSelector('imageNamed:inBundle:')).toEqual(['imageNamed']);
- });
- it('"play:by:" → ["play"]', () => {
- expect(swiftBaseNamesForObjcSelector('play:by:')).toEqual(['play']);
- });
- it('"playWithSong:by:" → ["playWithSong", "play"]', () => {
- expect(swiftBaseNamesForObjcSelector('playWithSong:by:').sort()).toEqual(
- ['play', 'playWithSong'].sort()
- );
- });
- it('"initWithName:" → includes "init"', () => {
- expect(swiftBaseNamesForObjcSelector('initWithName:')).toContain('init');
- });
- it('"initWithName:age:" → includes "init"', () => {
- expect(swiftBaseNamesForObjcSelector('initWithName:age:')).toContain('init');
- });
- it('"setName:" → includes the property name "name"', () => {
- expect(swiftBaseNamesForObjcSelector('setName:')).toContain('name');
- });
- it('"tableView:didSelectRowAtIndexPath:" → ["tableView"]', () => {
- expect(
- swiftBaseNamesForObjcSelector('tableView:didSelectRowAtIndexPath:')
- ).toEqual(['tableView']);
- });
- });
- describe('Source-window attribute detection', () => {
- it('detects literal @objc(custom)', () => {
- expect(detectExplicitObjcName(' @objc(custom:)\n func foo() {}')).toBe(
- 'custom:'
- );
- });
- it('returns null for plain @objc', () => {
- expect(detectExplicitObjcName('@objc func foo() {}')).toBeNull();
- });
- it('returns null when no @objc at all', () => {
- expect(detectExplicitObjcName('public func foo() {}')).toBeNull();
- });
- it('isObjcExposed true for @objc', () => {
- expect(isObjcExposed('@objc func foo() {}')).toBe(true);
- });
- it('isObjcExposed true for @objc(custom)', () => {
- expect(isObjcExposed('@objc(custom:) func foo() {}')).toBe(true);
- });
- it('isObjcExposed false for no annotation', () => {
- expect(isObjcExposed('public func foo() {}')).toBe(false);
- });
- it('@nonobjc opts out even if @objc also present (e.g. inside @objcMembers class)', () => {
- expect(isObjcExposed('@nonobjc @objc func foo() {}')).toBe(false);
- });
- });
|