From 18218938919f00672695deb4bad2c1f08ab333f9 Mon Sep 17 00:00:00 2001 From: Johannes Lumpe Date: Fri, 8 May 2015 20:05:37 +0300 Subject: [PATCH] Initial commit --- .gitignore | 2 + .jshintrc | 79 ++++++ FS.android.js | 11 + FS.ios.js | 58 ++++ LICENSE | 21 ++ NSArray+Map.h | 13 + NSArray+Map.m | 17 ++ README.md | 83 ++++++ RNFS.xcodeproj/project.pbxproj | 262 ++++++++++++++++++ .../contents.xcworkspacedata | 7 + .../xcshareddata/RNFS.xccheckout | 41 +++ .../UserInterfaceState.xcuserstate | Bin 0 -> 67643 bytes .../xcschemes/RNKeyboardEvents.xcscheme | 86 ++++++ .../xcschemes/xcschememanagement.plist | 27 ++ RNFSManager.h | 14 + RNFSManager.m | 105 +++++++ package.json | 27 ++ 17 files changed, 853 insertions(+) create mode 100644 .gitignore create mode 100644 .jshintrc create mode 100644 FS.android.js create mode 100644 FS.ios.js create mode 100644 LICENSE create mode 100644 NSArray+Map.h create mode 100644 NSArray+Map.m create mode 100644 README.md create mode 100644 RNFS.xcodeproj/project.pbxproj create mode 100644 RNFS.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 RNFS.xcodeproj/project.xcworkspace/xcshareddata/RNFS.xccheckout create mode 100644 RNFS.xcodeproj/project.xcworkspace/xcuserdata/johanneslumpe.xcuserdatad/UserInterfaceState.xcuserstate create mode 100644 RNFS.xcodeproj/xcuserdata/johanneslumpe.xcuserdatad/xcschemes/RNKeyboardEvents.xcscheme create mode 100644 RNFS.xcodeproj/xcuserdata/johanneslumpe.xcuserdatad/xcschemes/xcschememanagement.plist create mode 100644 RNFSManager.h create mode 100644 RNFSManager.m create mode 100644 package.json diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..5ee1d90 --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +node_modules +workbench diff --git a/.jshintrc b/.jshintrc new file mode 100644 index 0000000..b338af5 --- /dev/null +++ b/.jshintrc @@ -0,0 +1,79 @@ +{ + "-W093": true, + "asi": false, + "bitwise": true, + "boss": false, + "browser": false, + "camelcase": true, + "couch": false, + "curly": true, + "debug": false, + "devel": true, + "dojo": false, + "eqeqeq": true, + "eqnull": false, + "esnext": true, + "evil": false, + "expr": true, + "forin": false, + "freeze": true, + "funcscope": true, + "gcl": false, + "globalstrict": true, + "immed": false, + "indent": 2, + "iterator": false, + "jquery": false, + "lastsemic": false, + "latedef": false, + "laxbreak": true, + "laxcomma": false, + "loopfunc": false, + "maxcomplexity": false, + "maxdepth": false, + "maxerr": 50, + "maxlen": 80, + "maxparams": false, + "maxstatements": false, + "mootools": false, + "moz": false, + "multistr": false, + "newcap": true, + "noarg": true, + "node": true, + "noempty": true, + "nonbsp": true, + "nonew": true, + "nonstandard": false, + "notypeof": false, + "noyield": false, + "phantom": false, + "plusplus": false, + "predef": [ + "jasmine", + "describe", + "beforeEach", + "it", + "jest", + "pit", + "expect", + "rootRequire" + ], + "proto": false, + "prototypejs": false, + "quotmark": true, + "rhino": false, + "scripturl": false, + "shadow": false, + "smarttabs": false, + "strict": true, + "sub": false, + "supernew": false, + "trailing": true, + "undef": true, + "unused": true, + "validthis": false, + "worker": false, + "wsh": false, + "yui": false +} diff --git a/FS.android.js b/FS.android.js new file mode 100644 index 0000000..017ee25 --- /dev/null +++ b/FS.android.js @@ -0,0 +1,11 @@ +'use strict'; + +var warning = require('warning'); + +var FS = { + test: function() { + warning("Not yet implemented for Android."); + } +}; + +module.exports = FS; diff --git a/FS.ios.js b/FS.ios.js new file mode 100644 index 0000000..3052d74 --- /dev/null +++ b/FS.ios.js @@ -0,0 +1,58 @@ +'use strict'; + +var RNFSManager = require('NativeModules').RNFSManager; +var Promise = require('bluebird'); +var base64 = require('base-64'); + +var _readDir = Promise.promisify(RNFSManager.readDir); +var _stat = Promise.promisify(RNFSManager.stat); +var _readFile = Promise.promisify(RNFSManager.readFile); + +var convertError = (err) => { + var error = new Error(err.description); + error.code = err.code; + throw error; +}; + +var NSFileTypeRegular = RNFSManager.NSFileTypeRegular; +var NSFileTypeDirectory = RNFSManager.NSFileTypeDirectory; + +var RNFS = { + + readDir(path, rootDir) { + return _readDir(path, rootDir) + .catch(convertError); + }, + + stat(filepath) { + return _stat(filepath) + .then((result) => { + return { + 'ctime': new Date(result.ctime*1000), + 'mtime': new Date(result.mtime*1000), + 'size': result.size, + isFile: () => result.type === NSFileTypeRegular, + isDirectory: () => result.type === NSFileTypeDirectory, + }; + }) + .catch(convertError); + }, + + readFile(filepath, shouldDecode) { + var p = _readFile(filepath); + + if (shouldDecode !== false) { + p = p.then((data) => { + return base64.decode(data); + }); + } + + return p.catch(convertError); + }, + + MainBundle: RNFSManager.MainBundleDirectory, + CachesDirectory: RNFSManager.NSCachesDirectory, + DocumentDirectory: RNFSManager.NSDocumentDirectory +}; + +module.exports = RNFS; diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..aef1745 --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2015 Johannes Lumpe + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/NSArray+Map.h b/NSArray+Map.h new file mode 100644 index 0000000..ff3c960 --- /dev/null +++ b/NSArray+Map.h @@ -0,0 +1,13 @@ +// +// NSArray+Map.h +// RNFS +// +// taken from http://stackoverflow.com/questions/6127638/nsarray-equivalent-of-map + +#import + +@interface NSArray (Map) + +- (NSArray *)mapObjectsUsingBlock:(id (^)(id obj, NSUInteger idx))block; + +@end \ No newline at end of file diff --git a/NSArray+Map.m b/NSArray+Map.m new file mode 100644 index 0000000..fa4effb --- /dev/null +++ b/NSArray+Map.m @@ -0,0 +1,17 @@ +// +// NSArray+Map.m +// RNFS + +#import "NSArray+Map.h" + +@implementation NSArray (Map) + +- (NSArray *)mapObjectsUsingBlock:(id (^)(id obj, NSUInteger idx))block { + NSMutableArray *result = [NSMutableArray arrayWithCapacity:[self count]]; + [self enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) { + [result addObject:block(obj, idx)]; + }]; + return result; +} + +@end \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..d861ec0 --- /dev/null +++ b/README.md @@ -0,0 +1,83 @@ +## react-native-fs + +Native filesystem access for react-native + +Note: this project is under development and functionality will improve over time. Currently it provides only the bare minimum of functionality. + +Renaming, copying, and creating files will follow soon. + +## Usage + +First you need to install react-native-fs: + +```javascript +npm install react-native-fs --save +``` + +In XCode, in the project navigator, right click Libraries ➜ Add Files to [your project's name] Go to node_modules ➜ react-native-keyboardevents and add the .xcodeproj file + +In XCode, in the project navigator, select your project. Add the lib*.a from the RNFS project to your project's Build Phases ➜ Link Binary With Libraries Click .xcodeproj file you added before in the project navigator and go the Build Settings tab. Make sure 'All' is toggled on (instead of 'Basic'). Look for Header Search Paths and make sure it contains both $(SRCROOT)/../react-native/React and $(SRCROOT)/../../React - mark both as recursive. + +Run your project (Cmd+R) + +## Examples + +### Basic + +```javascript +// require the module +var RNFS = require('react-native-fs'); + +// get a list of files and directories in the main bundle +RNFS.readDir('/', RNFS.MainBundle) +.then((result) => { + console.log('GOT RESULT', result); + + // stat the first file + return Promise.all([RNFS.stat(result[0].path), result[0].path]); +}) +.then((statResult) => { + if (statResult[0].isFile()) { + // if we have a file, read it + return RNFS.readFile(statResult[1]); + } + + return 'no file'; +}) +.then((contents) => { + // log the file contents + console.log(contents); +}) +.catch((err) => { + console.log(err.message, err.code); +}); +``` + +## API + +### `promise readDir(path, directory)` + +Reads the contents of `path` in `directory`. +`path` is a string and `directory` is one of the following: +`RNFS.MainBundle`, `RNFS.CachesDirectory`, `RNFS.DocumentDirectory` + +The returned promise resolves with an array of objects with the following properties: + +`name` (`String`), The name of the item +`path` (`String`), The absolute path to the item + +### `promise stat(path)` + +Stats an item at `path`. +The promise resolves with an object with the following properties: +`ctime` (`Date`) - The creation date of the item +`mtime` (`Date`) - The modification date of the item +`size` (`Number`) - The size of the item in bytes +`isFile` (`Function`) - Returns true when the item is a file +`isDirectory` (`Function`) - Returns true when the item is a directory + +### `promise readFile(path, shouldDecode)` + +Reads the file at `path` and - by default - decodes the transferred base64 string. If `shouldDecode` is `false`, the base64 encoded string is returned + +Note: you will take quite a performance hit if you are reading big files diff --git a/RNFS.xcodeproj/project.pbxproj b/RNFS.xcodeproj/project.pbxproj new file mode 100644 index 0000000..349d01a --- /dev/null +++ b/RNFS.xcodeproj/project.pbxproj @@ -0,0 +1,262 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 46; + objects = { + +/* Begin PBXBuildFile section */ + F1E59BDF1ADD662800ACA28A /* RNFSManager.m in Sources */ = {isa = PBXBuildFile; fileRef = F1E59BDE1ADD662800ACA28A /* RNFSManager.m */; }; + F1EB08BB1AFD0E6A008F8F2B /* NSArray+Map.m in Sources */ = {isa = PBXBuildFile; fileRef = F1EB08BA1AFD0E6A008F8F2B /* NSArray+Map.m */; }; +/* End PBXBuildFile section */ + +/* Begin PBXCopyFilesBuildPhase section */ + F12AFB991ADAF8F800E0535D /* CopyFiles */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = "include/$(PRODUCT_NAME)"; + dstSubfolderSpec = 16; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + +/* Begin PBXFileReference section */ + F12AFB9B1ADAF8F800E0535D /* libRNFS.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = libRNFS.a; sourceTree = BUILT_PRODUCTS_DIR; }; + F1E59BDD1ADD662800ACA28A /* RNFSManager.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = RNFSManager.h; sourceTree = ""; }; + F1E59BDE1ADD662800ACA28A /* RNFSManager.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = RNFSManager.m; sourceTree = ""; }; + F1EB08B91AFD0E6A008F8F2B /* NSArray+Map.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = "NSArray+Map.h"; sourceTree = ""; }; + F1EB08BA1AFD0E6A008F8F2B /* NSArray+Map.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = "NSArray+Map.m"; sourceTree = ""; }; +/* End PBXFileReference section */ + +/* Begin PBXFrameworksBuildPhase section */ + F12AFB981ADAF8F800E0535D /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + F12AFB921ADAF8F800E0535D = { + isa = PBXGroup; + children = ( + F1EB08B91AFD0E6A008F8F2B /* NSArray+Map.h */, + F1EB08BA1AFD0E6A008F8F2B /* NSArray+Map.m */, + F1E59BDD1ADD662800ACA28A /* RNFSManager.h */, + F1E59BDE1ADD662800ACA28A /* RNFSManager.m */, + F12AFB9C1ADAF8F800E0535D /* Products */, + ); + sourceTree = ""; + }; + F12AFB9C1ADAF8F800E0535D /* Products */ = { + isa = PBXGroup; + children = ( + F12AFB9B1ADAF8F800E0535D /* libRNFS.a */, + ); + name = Products; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXNativeTarget section */ + F12AFB9A1ADAF8F800E0535D /* RNFS */ = { + isa = PBXNativeTarget; + buildConfigurationList = F12AFBAF1ADAF8F800E0535D /* Build configuration list for PBXNativeTarget "RNFS" */; + buildPhases = ( + F12AFB971ADAF8F800E0535D /* Sources */, + F12AFB981ADAF8F800E0535D /* Frameworks */, + F12AFB991ADAF8F800E0535D /* CopyFiles */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = RNFS; + productName = RNLocalNotification; + productReference = F12AFB9B1ADAF8F800E0535D /* libRNFS.a */; + productType = "com.apple.product-type.library.static"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + F12AFB931ADAF8F800E0535D /* Project object */ = { + isa = PBXProject; + attributes = { + LastUpgradeCheck = 0630; + ORGANIZATIONNAME = "Johannes Lumpe"; + TargetAttributes = { + F12AFB9A1ADAF8F800E0535D = { + CreatedOnToolsVersion = 6.3; + }; + }; + }; + buildConfigurationList = F12AFB961ADAF8F800E0535D /* Build configuration list for PBXProject "RNFS" */; + compatibilityVersion = "Xcode 3.2"; + developmentRegion = English; + hasScannedForEncodings = 0; + knownRegions = ( + en, + ); + mainGroup = F12AFB921ADAF8F800E0535D; + productRefGroup = F12AFB9C1ADAF8F800E0535D /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + F12AFB9A1ADAF8F800E0535D /* RNFS */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXSourcesBuildPhase section */ + F12AFB971ADAF8F800E0535D /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + F1E59BDF1ADD662800ACA28A /* RNFSManager.m in Sources */, + F1EB08BB1AFD0E6A008F8F2B /* NSArray+Map.m in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin XCBuildConfiguration section */ + F12AFBAD1ADAF8F800E0535D /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; + CLANG_CXX_LIBRARY = "libc++"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + ENABLE_STRICT_OBJC_MSGSEND = YES; + GCC_C_LANGUAGE_STANDARD = gnu99; + GCC_DYNAMIC_NO_PIC = NO; + GCC_NO_COMMON_BLOCKS = YES; + GCC_OPTIMIZATION_LEVEL = 0; + GCC_PREPROCESSOR_DEFINITIONS = ( + "DEBUG=1", + "$(inherited)", + ); + GCC_SYMBOLS_PRIVATE_EXTERN = NO; + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNDECLARED_SELECTOR = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + IPHONEOS_DEPLOYMENT_TARGET = 7.0; + MTL_ENABLE_DEBUG_INFO = YES; + ONLY_ACTIVE_ARCH = YES; + SDKROOT = iphoneos; + }; + name = Debug; + }; + F12AFBAE1ADAF8F800E0535D /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; + CLANG_CXX_LIBRARY = "libc++"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + ENABLE_NS_ASSERTIONS = NO; + ENABLE_STRICT_OBJC_MSGSEND = YES; + GCC_C_LANGUAGE_STANDARD = gnu99; + GCC_NO_COMMON_BLOCKS = YES; + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNDECLARED_SELECTOR = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + IPHONEOS_DEPLOYMENT_TARGET = 7.0; + MTL_ENABLE_DEBUG_INFO = NO; + SDKROOT = iphoneos; + VALIDATE_PRODUCT = YES; + }; + name = Release; + }; + F12AFBB01ADAF8F800E0535D /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + HEADER_SEARCH_PATHS = ( + "$(inherited)", + /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/include, + "$(SRCROOT)/../../React/**", + "$(SRCROOT)/../react-native/React/**", + "$(SRCROOT)/node_modules/react-native/React/**", + ); + OTHER_LDFLAGS = "-ObjC"; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + }; + name = Debug; + }; + F12AFBB11ADAF8F800E0535D /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + HEADER_SEARCH_PATHS = ( + "$(inherited)", + /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/include, + "$(SRCROOT)/../../React/**", + "$(SRCROOT)/../react-native/React/**", + "$(SRCROOT)/node_modules/react-native/React/**", + ); + OTHER_LDFLAGS = "-ObjC"; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + }; + name = Release; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + F12AFB961ADAF8F800E0535D /* Build configuration list for PBXProject "RNFS" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + F12AFBAD1ADAF8F800E0535D /* Debug */, + F12AFBAE1ADAF8F800E0535D /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + F12AFBAF1ADAF8F800E0535D /* Build configuration list for PBXNativeTarget "RNFS" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + F12AFBB01ADAF8F800E0535D /* Debug */, + F12AFBB11ADAF8F800E0535D /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + }; + rootObject = F12AFB931ADAF8F800E0535D /* Project object */; +} diff --git a/RNFS.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/RNFS.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 0000000..64e2374 --- /dev/null +++ b/RNFS.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/RNFS.xcodeproj/project.xcworkspace/xcshareddata/RNFS.xccheckout b/RNFS.xcodeproj/project.xcworkspace/xcshareddata/RNFS.xccheckout new file mode 100644 index 0000000..3713373 --- /dev/null +++ b/RNFS.xcodeproj/project.xcworkspace/xcshareddata/RNFS.xccheckout @@ -0,0 +1,41 @@ + + + + + IDESourceControlProjectFavoriteDictionaryKey + + IDESourceControlProjectIdentifier + D7886F59-B62A-41B8-8825-9080D60A1419 + IDESourceControlProjectName + RNFS + IDESourceControlProjectOriginsDictionary + + DBE4178F9C07984232B2B4DF265B9CB05D770D92 + https://github.com/johanneslumpe/react-native-keyboardevents.git + + IDESourceControlProjectPath + RNFS.xcodeproj + IDESourceControlProjectRelativeInstallPathDictionary + + DBE4178F9C07984232B2B4DF265B9CB05D770D92 + ../.. + + IDESourceControlProjectURL + https://github.com/johanneslumpe/react-native-keyboardevents.git + IDESourceControlProjectVersion + 111 + IDESourceControlProjectWCCIdentifier + DBE4178F9C07984232B2B4DF265B9CB05D770D92 + IDESourceControlProjectWCConfigurations + + + IDESourceControlRepositoryExtensionIdentifierKey + public.vcs.git + IDESourceControlWCCIdentifierKey + DBE4178F9C07984232B2B4DF265B9CB05D770D92 + IDESourceControlWCCName + react-native-fs + + + + diff --git a/RNFS.xcodeproj/project.xcworkspace/xcuserdata/johanneslumpe.xcuserdatad/UserInterfaceState.xcuserstate b/RNFS.xcodeproj/project.xcworkspace/xcuserdata/johanneslumpe.xcuserdatad/UserInterfaceState.xcuserstate new file mode 100644 index 0000000000000000000000000000000000000000..56165f180abbad8ad32b1e497c19a6732ad8d11b GIT binary patch literal 67643 zcmeEv2Y3@l)Bo-6-ATGT=?p291TbJqLhpt&Tx4)?i)90*L@~Ai2V*-f5CXipN$j5JmnCry&3OT|)&w2xFOg`_g6T$(G*ljchm(gJCr zR4FZymPz$egLHt@C^bo|r2j|UCVA}cd`fBL+oMpG<$|U%U)%#vDeuf>^t^7 z`+@z)equkfU)ZngH}*ScoO3Vt@$LBbdvj_`&=Tegr>~pUO|;r}H!TnS2x9 z%(w8Z{9=9?zlLASZ{~OK2l#{hA^sG9nm@x|;IHx5`8)g*{we>Af5*S)Kkz?>&Dy{$ijQDn^J=Vw{*IW{Z5Wx0oXeM4>1W#iB&)BPzrKu~4iRAPx~nh$F?Z;y7`V zI9Z%7&JgE_bH#GI z;#KjQcwKxfJ`taa&%{sSXYq^pRs1G?7k`L9WuL6cx@^dPIUt*In!K|-KprR$l1Is- z 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 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/RNFS.xcodeproj/xcuserdata/johanneslumpe.xcuserdatad/xcschemes/xcschememanagement.plist b/RNFS.xcodeproj/xcuserdata/johanneslumpe.xcuserdatad/xcschemes/xcschememanagement.plist new file mode 100644 index 0000000..d0ec77b --- /dev/null +++ b/RNFS.xcodeproj/xcuserdata/johanneslumpe.xcuserdatad/xcschemes/xcschememanagement.plist @@ -0,0 +1,27 @@ + + + + + SchemeUserState + + RNKeyboardEvents.xcscheme + + orderHint + 1 + + + SuppressBuildableAutocreation + + F12AFB9A1ADAF8F800E0535D + + primary + + + F12AFBA51ADAF8F800E0535D + + primary + + + + + diff --git a/RNFSManager.h b/RNFSManager.h new file mode 100644 index 0000000..c801aae --- /dev/null +++ b/RNFSManager.h @@ -0,0 +1,14 @@ +// +// RNFSManager.h +// RNFSManager +// +// Created by Johannes Lumpe on 08/05/15. +// Copyright (c) 2015 Johannes Lumpe. All rights reserved. +// + +#import "RCTBridgeModule.h" +#import "RCTLog.h" + +@interface RNFSManager : NSObject + +@end diff --git a/RNFSManager.m b/RNFSManager.m new file mode 100644 index 0000000..38a10df --- /dev/null +++ b/RNFSManager.m @@ -0,0 +1,105 @@ +// +// RNFSManager.m +// RNFSManager +// +// Created by Johannes Lumpe on 08/05/15. +// Copyright (c) 2015 Johannes Lumpe. All rights reserved. +// + +#import "RNFSManager.h" +#import "RCTConvert.h" +#import "RCTBridge.h" +#import "NSArray+Map.h" + +@implementation RNFSManager + +static int MainBundleDirectory = 999; + +@synthesize bridge = _bridge; +RCT_EXPORT_MODULE(); + + +RCT_EXPORT_METHOD(readDir:(NSString*)directory inFolder:(NSNumber*)folder callback:(RCTResponseSenderBlock)callback){ + NSString *path; + int folderInt = [folder integerValue]; + //NSLog(@"%d %d", folderInt, MainBundleDirectory); + if (folderInt == MainBundleDirectory) { + path = [[NSBundle mainBundle] bundlePath]; + } else { + NSArray *paths = NSSearchPathForDirectoriesInDomains(folderInt, NSUserDomainMask, YES); + path = [paths objectAtIndex:0]; + } + + NSFileManager *fileManager = [NSFileManager defaultManager]; + NSError *error; + NSString * dirPath = [path stringByAppendingPathComponent:directory]; + NSArray *contents = [fileManager contentsOfDirectoryAtPath:dirPath error:&error]; + + contents = [contents mapObjectsUsingBlock:^id(id obj, NSUInteger idx) { + return @{ + @"name": (NSString*)obj, + @"path": [dirPath stringByAppendingPathComponent:(NSString*)obj] + }; + }]; + + if (error) { + return callback([self makeErrorPayload:error]); + } + + callback(@[[NSNull null], contents]); +} + +RCT_EXPORT_METHOD(stat:(NSString*)filepath callback:(RCTResponseSenderBlock)callback){ + NSError *error; + NSDictionary *attributes = [[NSFileManager defaultManager] attributesOfItemAtPath:filepath error:&error]; + // NSLog(@"%@", attributes); + + if (error) { + return callback([self makeErrorPayload:error]); + } + + attributes = @{ + @"ctime": [self dateToTimeIntervalNumber:(NSDate*)[attributes objectForKey:NSFileCreationDate]], + @"mtime": [self dateToTimeIntervalNumber:(NSDate*)[attributes objectForKey:NSFileModificationDate]], + @"size": [attributes objectForKey:NSFileSize], + @"type": [attributes objectForKey:NSFileType], + @"mode": [NSNumber numberWithInteger:[[NSString stringWithFormat:@"%o", [(NSNumber*)[attributes objectForKey:NSFilePosixPermissions] integerValue]] integerValue]] + }; + + callback(@[[NSNull null], attributes]); +} + +RCT_EXPORT_METHOD(readFile:(NSString*)filepath callback:(RCTResponseSenderBlock)callback){ + NSData *content = [[NSFileManager defaultManager] contentsAtPath:filepath]; + NSString *base64Content = [content base64EncodedStringWithOptions:NSDataBase64EncodingEndLineWithLineFeed]; + NSLog(@"%@", base64Content); + if (!base64Content) { + return callback(@[[NSString stringWithFormat:@"Could not read file at path %@", filepath]]); + } + + callback(@[[NSNull null], base64Content]); +} + +- (NSNumber*) dateToTimeIntervalNumber:(NSDate*)date { + return [NSNumber numberWithDouble:[date timeIntervalSince1970]]; +} + +- (NSArray*) makeErrorPayload:(NSError*)error { + return @[@{ + @"description": error.localizedDescription, + @"code": [NSNumber numberWithInteger:error.code] + }]; +} + +- (NSDictionary *)constantsToExport +{ + return @{ + @"NSCachesDirectory": [NSNumber numberWithInteger:NSCachesDirectory], + @"NSDocumentDirectory": [NSNumber numberWithInteger:NSDocumentDirectory], + @"MainBundleDirectory": [NSNumber numberWithInteger:MainBundleDirectory], + @"NSFileTypeRegular": NSFileTypeRegular, + @"NSFileTypeDirectory": NSFileTypeDirectory + }; +} + +@end diff --git a/package.json b/package.json new file mode 100644 index 0000000..4840b15 --- /dev/null +++ b/package.json @@ -0,0 +1,27 @@ +{ + "name": "react-native-fs", + "version": "0.1.0", + "description": "Native filesystem access for react-native", + "main": "FS.ios.js", + "scripts": { + "test": "echo \"Error: no test specified\" && exit 1" + }, + "repository": { + "type": "git", + "url": "git@github.com:johanneslumpe/react-native-fs.git" + }, + "keywords": [ + "react-component", + "react-native", + "ios", + "fs", + "filesystem" + ], + "author": "Johannes Lumpe (https://github.com/johanneslumpe)", + "license": "MIT", + "dependencies": { + "base-64": "^0.1.0", + "bluebird": "^2.9.25", + "react-native": ">=0.3.4 <0.5.0" + } +}