diff --git a/CHANGELOG b/CHANGELOG index 83072a1e9a..c79a097717 100644 --- a/CHANGELOG +++ b/CHANGELOG @@ -1,5 +1,4 @@ -cocos2d-x-3.3 ?? - [NEW] Added submesh support, and support attaching a Sprite3D object to a bone +cocos2d-x-3.3alpha0 ?? [NEW] Added UIScale9Sprite [NEW] Added Camera, AABB, OBB and Ray [NEW] Added render primitive and render primitive command, support passing point, line and triangle data diff --git a/cocos/2d/CCAction.h b/cocos/2d/CCAction.h index f23a1f6fa1..35319d1f05 100644 --- a/cocos/2d/CCAction.h +++ b/cocos/2d/CCAction.h @@ -107,10 +107,11 @@ public: inline int getTag() const { return _tag; } inline void setTag(int tag) { _tag = tag; } -protected: +CC_CONSTRUCTOR_ACCESS: Action(); virtual ~Action(); +protected: Node *_originalTarget; /** The "target". The target will be set with the 'startWithTarget' method. @@ -156,12 +157,13 @@ public: return nullptr; } -protected: +CC_CONSTRUCTOR_ACCESS: FiniteTimeAction() : _duration(0) {} virtual ~FiniteTimeAction(){} +protected: //! duration in seconds float _duration; diff --git a/cocos/2d/CCActionGrid.h b/cocos/2d/CCActionGrid.h index 30b53c3dd9..544db56ed4 100644 --- a/cocos/2d/CCActionGrid.h +++ b/cocos/2d/CCActionGrid.h @@ -56,12 +56,13 @@ public: virtual GridAction* reverse() const override; virtual void startWithTarget(Node *target) override; -protected: +CC_CONSTRUCTOR_ACCESS: GridAction() {} virtual ~GridAction() {} /** initializes the action with size and duration */ bool initWithDuration(float duration, const Size& gridSize); +protected: Size _gridSize; NodeGrid* _gridNodeTarget; @@ -285,10 +286,11 @@ public: virtual StopGrid* clone() const override; virtual StopGrid* reverse() const override; -protected: +CC_CONSTRUCTOR_ACCESS: StopGrid() {} virtual ~StopGrid() {} +protected: NodeGrid* _gridNodeTarget; void cacheTargetAsGridNode(); diff --git a/cocos/2d/CCActionGrid3D.h b/cocos/2d/CCActionGrid3D.h index 89c187cc58..4c9491e3c0 100644 --- a/cocos/2d/CCActionGrid3D.h +++ b/cocos/2d/CCActionGrid3D.h @@ -108,7 +108,7 @@ public: virtual void update(float time) override; virtual FlipY3D* clone() const override; -protected: +CC_CONSTRUCTOR_ACCESS: FlipY3D() {} virtual ~FlipY3D() {} private: diff --git a/cocos/2d/CCActionInstant.h b/cocos/2d/CCActionInstant.h index b175e54b8b..e5428fc130 100644 --- a/cocos/2d/CCActionInstant.h +++ b/cocos/2d/CCActionInstant.h @@ -412,13 +412,14 @@ public: virtual __CCCallFuncND* clone() const override; virtual void execute() override; -protected: +CC_CONSTRUCTOR_ACCESS: __CCCallFuncND() {} virtual ~__CCCallFuncND() {} /** initializes the action with the callback and the data to pass as an argument */ bool initWithTarget(Ref* target, SEL_CallFuncND selector, void* d); +protected: SEL_CallFuncND _callFuncND; void* _data; @@ -451,7 +452,7 @@ public: Ref* getObject() const; void setObject(Ref* obj); -protected: +CC_CONSTRUCTOR_ACCESS: __CCCallFuncO(); virtual ~__CCCallFuncO(); /** initializes the action with the callback @@ -460,7 +461,7 @@ protected: */ bool initWithTarget(Ref* target, SEL_CallFuncO selector, Ref* object); - +protected: /** object to be passed as argument */ Ref* _object; SEL_CallFuncO _callFuncO; diff --git a/cocos/2d/CCActionTiledGrid.h b/cocos/2d/CCActionTiledGrid.h index 6f3e846353..844e814b70 100644 --- a/cocos/2d/CCActionTiledGrid.h +++ b/cocos/2d/CCActionTiledGrid.h @@ -142,7 +142,7 @@ public: virtual void update(float time) override; virtual FadeOutTRTiles* clone() const override; -protected: +CC_CONSTRUCTOR_ACCESS: FadeOutTRTiles() {} virtual ~FadeOutTRTiles() {} @@ -163,7 +163,7 @@ public: virtual float testFunc(const Size& pos, float time) override; virtual FadeOutBLTiles* clone() const override; -protected: +CC_CONSTRUCTOR_ACCESS: FadeOutBLTiles() {} virtual ~FadeOutBLTiles() {} @@ -186,7 +186,7 @@ public: virtual FadeOutUpTiles* clone() const override; virtual float testFunc(const Size& pos, float time) override; -protected: +CC_CONSTRUCTOR_ACCESS: FadeOutUpTiles() {} virtual ~FadeOutUpTiles() {} @@ -207,7 +207,7 @@ public: virtual FadeOutDownTiles* clone() const override; virtual float testFunc(const Size& pos, float time) override; -protected: +CC_CONSTRUCTOR_ACCESS: FadeOutDownTiles() {} virtual ~FadeOutDownTiles() {} diff --git a/cocos/2d/CCLabel.h b/cocos/2d/CCLabel.h index a81daf8e6b..de4173527c 100644 --- a/cocos/2d/CCLabel.h +++ b/cocos/2d/CCLabel.h @@ -275,6 +275,18 @@ public: CC_DEPRECATED_ATTRIBUTE int getCommonLineHeight() const { return getLineHeight();} +CC_CONSTRUCTOR_ACCESS: + /** + * @js NA + */ + Label(FontAtlas *atlas = nullptr, TextHAlignment hAlignment = TextHAlignment::LEFT, + TextVAlignment vAlignment = TextVAlignment::TOP,bool useDistanceField = false,bool useA8Shader = false); + /** + * @js NA + * @lua NA + */ + virtual ~Label(); + protected: void onDraw(const Mat4& transform, bool transformUpdated); @@ -294,17 +306,6 @@ protected: STRING_TEXTURE }; - /** - * @js NA - */ - Label(FontAtlas *atlas = nullptr, TextHAlignment hAlignment = TextHAlignment::LEFT, - TextVAlignment vAlignment = TextVAlignment::TOP,bool useDistanceField = false,bool useA8Shader = false); - /** - * @js NA - * @lua NA - */ - virtual ~Label(); - virtual void setFontAtlas(FontAtlas* atlas,bool distanceFieldEnabled = false, bool useA8Shader = false); bool recordLetterInfo(const cocos2d::Vec2& point,const FontLetterDefinition& letterDef, int spriteIndex); diff --git a/cocos/2d/CCLabelAtlas.h b/cocos/2d/CCLabelAtlas.h index 13db8b0ed1..b0e26d3ffb 100644 --- a/cocos/2d/CCLabelAtlas.h +++ b/cocos/2d/CCLabelAtlas.h @@ -96,6 +96,8 @@ CC_CONSTRUCTOR_ACCESS: { _string.clear(); } + +protected: virtual void updateColor() override; #if CC_LABELATLAS_DEBUG_DRAW diff --git a/cocos/2d/CCLayer.h b/cocos/2d/CCLayer.h index faf9817137..08abcb3201 100644 --- a/cocos/2d/CCLayer.h +++ b/cocos/2d/CCLayer.h @@ -228,7 +228,7 @@ public: virtual void setOpacityModifyRGB(bool bValue) override { return Layer::setOpacityModifyRGB(bValue); } virtual bool isOpacityModifyRGB() const override { return Layer::isOpacityModifyRGB(); } -protected: +CC_CONSTRUCTOR_ACCESS: __LayerRGBA(); virtual ~__LayerRGBA() {} diff --git a/cocos/2d/CCNodeGrid.h b/cocos/2d/CCNodeGrid.h index 36671ae7b0..6465d5a8e3 100644 --- a/cocos/2d/CCNodeGrid.h +++ b/cocos/2d/CCNodeGrid.h @@ -56,10 +56,11 @@ public: // overrides virtual void visit(Renderer *renderer, const Mat4 &parentTransform, uint32_t parentFlags) override; -protected: +CC_CONSTRUCTOR_ACCESS: NodeGrid(); virtual ~NodeGrid(); +protected: void onGridBeginDraw(); void onGridEndDraw(); diff --git a/cocos/2d/cocos2d_winrt.vcxproj b/cocos/2d/cocos2d_winrt.vcxproj index 3ad2b3a2f5..61a777d5a6 100644 --- a/cocos/2d/cocos2d_winrt.vcxproj +++ b/cocos/2d/cocos2d_winrt.vcxproj @@ -315,6 +315,7 @@ + @@ -345,6 +346,7 @@ CompileAsCpp + @@ -423,6 +425,8 @@ + + @@ -430,6 +434,8 @@ + + @@ -507,6 +513,7 @@ + @@ -535,6 +542,7 @@ + @@ -627,6 +635,8 @@ + + @@ -635,6 +645,8 @@ + + diff --git a/cocos/2d/cocos2d_winrt.vcxproj.filters b/cocos/2d/cocos2d_winrt.vcxproj.filters index 1fc613a19b..c49292bf34 100644 --- a/cocos/2d/cocos2d_winrt.vcxproj.filters +++ b/cocos/2d/cocos2d_winrt.vcxproj.filters @@ -581,6 +581,24 @@ base + + base + + + base + + + renderer + + + renderer + + + renderer + + + renderer + @@ -1188,6 +1206,24 @@ base + + base + + + base + + + renderer + + + renderer + + + renderer + + + renderer + diff --git a/cocos/3d/CCBundle3D.h b/cocos/3d/CCBundle3D.h index 467015f272..6388e14ad6 100644 --- a/cocos/3d/CCBundle3D.h +++ b/cocos/3d/CCBundle3D.h @@ -200,10 +200,9 @@ protected: */ Reference* seekToFirstType(unsigned int type); -protected: CC_CONSTRUCTOR_ACCESS: Bundle3D(); - ~Bundle3D(); + virtual ~Bundle3D(); protected: diff --git a/cocos/Android.mk b/cocos/Android.mk index e18bfa6995..ef21816612 100644 --- a/cocos/Android.mk +++ b/cocos/Android.mk @@ -188,6 +188,7 @@ physics/chipmunk/CCPhysicsWorldInfo_chipmunk.cpp \ LOCAL_EXPORT_C_INCLUDES := $(LOCAL_PATH) \ $(LOCAL_PATH)/. \ $(LOCAL_PATH)/platform/android \ + $(LOCAL_PATH)/base \ $(LOCAL_PATH)/../external/tinyxml2 \ $(LOCAL_PATH)/../external/unzip \ $(LOCAL_PATH)/../external/chipmunk/include/chipmunk \ @@ -214,6 +215,9 @@ LOCAL_EXPORT_LDLIBS := -lGLESv2 \ LOCAL_STATIC_LIBRARIES := cocos_freetype2_static LOCAL_STATIC_LIBRARIES += chipmunk_static LOCAL_STATIC_LIBRARIES += cocos_png_static +LOCAL_STATIC_LIBRARIES += cocos_jpeg_static +LOCAL_STATIC_LIBRARIES += cocos_tiff_static +LOCAL_STATIC_LIBRARIES += cocos_webp_static LOCAL_WHOLE_STATIC_LIBRARIES := cocos2dxandroid_static # define the macro to compile through support/zip_support/ioapi.c @@ -228,3 +232,6 @@ $(call import-module,freetype2/prebuilt/android) $(call import-module,chipmunk) $(call import-module,platform/android) $(call import-module,png/prebuilt/android) +$(call import-module,jpeg/prebuilt/android) +$(call import-module,tiff/prebuilt/android) +$(call import-module,webp/prebuilt/android) diff --git a/cocos/base/CCEvent.h b/cocos/base/CCEvent.h index 32c7537c1e..fa2445d60a 100644 --- a/cocos/base/CCEvent.h +++ b/cocos/base/CCEvent.h @@ -55,7 +55,7 @@ public: CUSTOM }; -protected: +CC_CONSTRUCTOR_ACCESS: /** Constructor */ Event(Type type); public: diff --git a/cocos/base/CCEventListener.h b/cocos/base/CCEventListener.h index 47dad7cd07..18a135515f 100644 --- a/cocos/base/CCEventListener.h +++ b/cocos/base/CCEventListener.h @@ -63,7 +63,7 @@ public: typedef std::string ListenerID; -protected: +CC_CONSTRUCTOR_ACCESS: /** Constructor */ EventListener(); diff --git a/cocos/base/CCEventListenerAcceleration.h b/cocos/base/CCEventListenerAcceleration.h index daa4d6abe3..ea339770ca 100644 --- a/cocos/base/CCEventListenerAcceleration.h +++ b/cocos/base/CCEventListenerAcceleration.h @@ -41,10 +41,13 @@ public: /// Overrides virtual EventListenerAcceleration* clone() override; virtual bool checkAvailable() override; -private: + +CC_CONSTRUCTOR_ACCESS: EventListenerAcceleration(); bool init(const std::function& callback); + +private: std::function onAccelerationEvent; friend class LuaEventListenerAcceleration; diff --git a/cocos/base/CCEventListenerFocus.h b/cocos/base/CCEventListenerFocus.h index 2b0f61eb31..300fdcdac8 100644 --- a/cocos/base/CCEventListenerFocus.h +++ b/cocos/base/CCEventListenerFocus.h @@ -51,7 +51,7 @@ public: public: std::function onFocusChanged; -private: +CC_CONSTRUCTOR_ACCESS: EventListenerFocus(); bool init(); diff --git a/cocos/base/CCEventListenerKeyboard.h b/cocos/base/CCEventListenerKeyboard.h index decf898094..283a0ed41e 100644 --- a/cocos/base/CCEventListenerKeyboard.h +++ b/cocos/base/CCEventListenerKeyboard.h @@ -46,7 +46,7 @@ public: std::function onKeyPressed; std::function onKeyReleased; -private: +CC_CONSTRUCTOR_ACCESS: EventListenerKeyboard(); bool init(); }; diff --git a/cocos/base/CCEventListenerMouse.h b/cocos/base/CCEventListenerMouse.h index fb54c27bac..c5141444b7 100644 --- a/cocos/base/CCEventListenerMouse.h +++ b/cocos/base/CCEventListenerMouse.h @@ -49,7 +49,7 @@ public: std::function onMouseMove; std::function onMouseScroll; -private: +CC_CONSTRUCTOR_ACCESS: EventListenerMouse(); bool init(); }; diff --git a/cocos/base/CCEventListenerTouch.h b/cocos/base/CCEventListenerTouch.h index ff272511e9..b125b8e95f 100644 --- a/cocos/base/CCEventListenerTouch.h +++ b/cocos/base/CCEventListenerTouch.h @@ -56,10 +56,11 @@ public: std::function onTouchEnded; std::function onTouchCancelled; -private: +CC_CONSTRUCTOR_ACCESS: EventListenerTouchOneByOne(); bool init(); +private: std::vector _claimedTouches; bool _needSwallow; @@ -85,7 +86,7 @@ public: std::function&, Event*)> onTouchesEnded; std::function&, Event*)> onTouchesCancelled; -private: +CC_CONSTRUCTOR_ACCESS: EventListenerTouchAllAtOnce(); bool init(); private: diff --git a/cocos/base/ccConfig.h b/cocos/base/ccConfig.h index 2225998746..a44ede24ef 100644 --- a/cocos/base/ccConfig.h +++ b/cocos/base/ccConfig.h @@ -269,6 +269,24 @@ To enable set it to a value different than 0. Disabled by default. #define CC_USE_PHYSICS 1 #endif +/** Support JPEG or not. If your application don't use jpeg format picture, you can undefine this macro to save package size. + */ +#ifndef CC_USE_JPEG +#define CC_USE_JPEG 1 +#endif // CC_USE_JPEG + +/** Support TIFF or not. If your application don't use TIFF format picture, you can undefine this macro to save package size. + */ +#ifndef CC_USE_TIFF +#define CC_USE_TIFF 1 +#endif // CC_USE_TIFF + +/** Support webp or not. If your application don't use webp format picture, you can undefine this macro to save package size. + */ +#ifndef CC_USE_WEBP +#define CC_USE_WEBP 1 +#endif // CC_USE_WEBP + /** Enable Script binding */ #ifndef CC_ENABLE_SCRIPT_BINDING #define CC_ENABLE_SCRIPT_BINDING 1 diff --git a/cocos/cocos2d.cpp b/cocos/cocos2d.cpp index 83c40ab695..adc8a7ac05 100644 --- a/cocos/cocos2d.cpp +++ b/cocos/cocos2d.cpp @@ -31,7 +31,7 @@ NS_CC_BEGIN CC_DLL const char* cocos2dVersion() { - return "cocos2d-x 3.3"; + return "cocos2d-x 3.3alpha0"; } NS_CC_END diff --git a/cocos/editor-support/cocostudio/Android.mk b/cocos/editor-support/cocostudio/Android.mk index 39fecd8321..c6356d29c4 100644 --- a/cocos/editor-support/cocostudio/Android.mk +++ b/cocos/editor-support/cocostudio/Android.mk @@ -66,7 +66,8 @@ $(LOCAL_PATH)/../../../external LOCAL_C_INCLUDES := $(LOCAL_PATH)/../.. \ $(LOCAL_PATH)/../../../external \ $(LOCAL_PATH)/.. \ -$(LOCAL_PATH)/../.. +$(LOCAL_PATH)/../.. \ +$(LOCAL_PATH)/WidgetReader LOCAL_CFLAGS += -fexceptions diff --git a/cocos/editor-support/cocostudio/CCInputDelegate.h b/cocos/editor-support/cocostudio/CCInputDelegate.h index fb41d543d0..15b1233bb0 100644 --- a/cocos/editor-support/cocostudio/CCInputDelegate.h +++ b/cocos/editor-support/cocostudio/CCInputDelegate.h @@ -44,7 +44,7 @@ namespace cocostudio { */ class CC_STUDIO_DLL InputDelegate { -protected: +CC_CONSTRUCTOR_ACCESS: /** * @js ctor */ diff --git a/cocos/editor-support/cocostudio/CCSSceneReader.h b/cocos/editor-support/cocostudio/CCSSceneReader.h index e4fa36087d..1980b47ce0 100644 --- a/cocos/editor-support/cocostudio/CCSSceneReader.h +++ b/cocos/editor-support/cocostudio/CCSSceneReader.h @@ -65,9 +65,11 @@ public: void setTarget(const std::function& selector); cocos2d::Node* getNodeByTag(int nTag); inline AttachComponentType getAttachComponentType(){return _attachComponent;} -private: +CC_CONSTRUCTOR_ACCESS: SceneReader(void); virtual ~SceneReader(void); + +private: std::string getComponentClassName(const std::string& name); cocos2d::Component* createComponent(const std::string classname); diff --git a/cocos/platform/CCImage.cpp b/cocos/platform/CCImage.cpp index 80ba8e672b..d022327fc1 100644 --- a/cocos/platform/CCImage.cpp +++ b/cocos/platform/CCImage.cpp @@ -30,6 +30,7 @@ THE SOFTWARE. #include #include "base/CCData.h" +#include "base/ccConfig.h" // CC_USE_JPEG, CC_USE_TIFF, CC_USE_WEBP extern "C" { @@ -51,21 +52,31 @@ extern "C" { return strerror(errnum); } - // int fputs(const char * __restrict, FILE * __restrict) __DARWIN_ALIAS(fputs); - int fputs$UNIX2003(const char *str, FILE *stream) - { - return fputs(str, stream); - } #endif #endif #include "png.h" + +#if CC_USE_TIFF +#include "tiffio.h" +#endif //CC_USE_TIFF + #include "base/etc1.h" + +#if CC_USE_JPEG +#include "jpeglib.h" +#endif // CC_USE_JPEG } #include "base/s3tc.h" #include "base/atitc.h" #include "base/pvr.h" #include "base/TGAlib.h" +#if (CC_TARGET_PLATFORM != CC_PLATFORM_WP8) && (CC_TARGET_PLATFORM != CC_PLATFORM_WINRT) +#if CC_USE_WEBP +#include "decode.h" +#endif // CC_USE_WEBP +#endif + #include "base/ccMacros.h" #include "CCCommon.h" #include "CCStdC.h" @@ -73,7 +84,6 @@ extern "C" #include "base/CCConfiguration.h" #include "base/ccUtils.h" #include "base/ZipUtils.h" -#include "base/CCModuleManager.h" #if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID) #include "android/CCFileUtilsAndroid.h" #endif @@ -90,6 +100,7 @@ NS_CC_BEGIN namespace { static const int PVR_TEXTURE_FLAG_TYPE_MASK = 0xff; + static bool _PVRHaveAlphaPremultiplied = false; // Values taken from PVRTexture.h from http://www.imgtec.com @@ -752,35 +763,172 @@ bool Image::isCompressed() return Texture2D::getPixelFormatInfoMap().at(_renderFormat).compressed; } +namespace +{ +/* + * ERROR HANDLING: + * + * The JPEG library's standard error handler (jerror.c) is divided into + * several "methods" which you can override individually. This lets you + * adjust the behavior without duplicating a lot of code, which you might + * have to update with each future release. + * + * We override the "error_exit" method so that control is returned to the + * library's caller when a fatal error occurs, rather than calling exit() + * as the standard error_exit method does. + * + * We use C's setjmp/longjmp facility to return control. This means that the + * routine which calls the JPEG library must first execute a setjmp() call to + * establish the return point. We want the replacement error_exit to do a + * longjmp(). But we need to make the setjmp buffer accessible to the + * error_exit routine. To do this, we make a private extension of the + * standard JPEG error handler object. (If we were using C++, we'd say we + * were making a subclass of the regular error handler.) + * + * Here's the extended error handler struct: + */ +#if CC_USE_JPEG + struct MyErrorMgr + { + struct jpeg_error_mgr pub; /* "public" fields */ + jmp_buf setjmp_buffer; /* for return to caller */ + }; + + typedef struct MyErrorMgr * MyErrorPtr; + + /* + * Here's the routine that will replace the standard error_exit method: + */ + + METHODDEF(void) + myErrorExit(j_common_ptr cinfo) + { + /* cinfo->err really points to a MyErrorMgr struct, so coerce pointer */ + MyErrorPtr myerr = (MyErrorPtr) cinfo->err; + + /* Always display the message. */ + /* We could postpone this until after returning, if we chose. */ + /* internal message function cann't show error message in some platforms, so we rewrite it here. + * edit it if has version confilict. + */ + //(*cinfo->err->output_message) (cinfo); + char buffer[JMSG_LENGTH_MAX]; + (*cinfo->err->format_message) (cinfo, buffer); + CCLOG("jpeg error: %s", buffer); + + /* Return control to the setjmp point */ + longjmp(myerr->setjmp_buffer, 1); + } +#endif // CC_USE_JPEG +} + bool Image::initWithJpgData(const unsigned char * data, ssize_t dataLen) { - _hasPremultipliedAlpha = false; - - JPEGModule *jpegModule = static_cast(ModuleManager::getModule("jpeg")); - if (jpegModule) +#if CC_USE_JPEG + /* these are standard libjpeg structures for reading(decompression) */ + struct jpeg_decompress_struct cinfo; + /* We use our private extension JPEG error handler. + * Note that this struct must live as long as the main JPEG parameter + * struct, to avoid dangling-pointer problems. + */ + struct MyErrorMgr jerr; + /* libjpeg data structure for storing one row, that is, scanline of an image */ + JSAMPROW row_pointer[1] = {0}; + unsigned long location = 0; + unsigned int i = 0; + + bool ret = false; + do { - DataFromModule dataFromJPEGInit; - bool result = jpegModule->initWithJPEGData(data, dataLen, dataFromJPEGInit); - _width = dataFromJPEGInit.width; - _height = dataFromJPEGInit.height; - _renderFormat = dataFromJPEGInit.renderFormat; - _dataLen = dataFromJPEGInit.dataLength; - _data = dataFromJPEGInit.data; - - return result; - } - else + /* We set up the normal JPEG error routines, then override error_exit. */ + cinfo.err = jpeg_std_error(&jerr.pub); + jerr.pub.error_exit = myErrorExit; + /* Establish the setjmp return context for MyErrorExit to use. */ + if (setjmp(jerr.setjmp_buffer)) + { + /* If we get here, the JPEG code has signaled an error. + * We need to clean up the JPEG object, close the input file, and return. + */ + jpeg_destroy_decompress(&cinfo); + break; + } + + /* setup decompression process and source, then read JPEG header */ + jpeg_create_decompress( &cinfo ); + +#ifndef CC_TARGET_QT5 + jpeg_mem_src(&cinfo, const_cast(data), dataLen); +#endif /* CC_TARGET_QT5 */ + + /* reading the image header which contains image information */ +#if (JPEG_LIB_VERSION >= 90) + // libjpeg 0.9 adds stricter types. + jpeg_read_header(&cinfo, TRUE); +#else + jpeg_read_header(&cinfo, TRUE); +#endif + + // we only support RGB or grayscale + if (cinfo.jpeg_color_space == JCS_GRAYSCALE) + { + _renderFormat = Texture2D::PixelFormat::I8; + }else + { + cinfo.out_color_space = JCS_RGB; + _renderFormat = Texture2D::PixelFormat::RGB888; + } + + /* Start decompression jpeg here */ + jpeg_start_decompress( &cinfo ); + + /* init image info */ + _width = cinfo.output_width; + _height = cinfo.output_height; + _hasPremultipliedAlpha = false; + row_pointer[0] = static_cast(malloc(cinfo.output_width*cinfo.output_components * sizeof(unsigned char))); + CC_BREAK_IF(! row_pointer[0]); + + _dataLen = cinfo.output_width*cinfo.output_height*cinfo.output_components; + _data = static_cast(malloc(_dataLen * sizeof(unsigned char))); + CC_BREAK_IF(! _data); + + /* now actually read the jpeg into the raw buffer */ + /* read one scan line at a time */ + while (cinfo.output_scanline < cinfo.output_height) + { + jpeg_read_scanlines( &cinfo, row_pointer, 1 ); + for (i=0; i(malloc(_dataLen * sizeof(unsigned char))); - if(!_data) + if (!_data) { if (row_pointers != nullptr) { @@ -915,38 +1064,192 @@ bool Image::initWithPngData(const unsigned char * data, ssize_t dataLen) free(row_pointers); } - bRet = true; + ret = true; } while (0); if (png_ptr) { png_destroy_read_struct(&png_ptr, (info_ptr) ? &info_ptr : 0, 0); } - return bRet; + return ret; } +#if CC_USE_TIFF +namespace +{ + static tmsize_t tiffReadProc(thandle_t fd, void* buf, tmsize_t size) + { + tImageSource* isource = (tImageSource*)fd; + uint8* ma; + uint64 mb; + unsigned long n; + unsigned long o; + tmsize_t p; + ma=(uint8*)buf; + mb=size; + p=0; + while (mb>0) + { + n=0x80000000UL; + if ((uint64)n>mb) + n=(unsigned long)mb; + + + if ((int)(isource->offset + n) <= isource->size) + { + memcpy(ma, isource->data+isource->offset, n); + isource->offset += n; + o = n; + } + else + { + return 0; + } + + ma+=o; + mb-=o; + p+=o; + if (o!=n) + { + break; + } + } + return p; + } + + static tmsize_t tiffWriteProc(thandle_t fd, void* buf, tmsize_t size) + { + CC_UNUSED_PARAM(fd); + CC_UNUSED_PARAM(buf); + CC_UNUSED_PARAM(size); + return 0; + } + + + static uint64 tiffSeekProc(thandle_t fd, uint64 off, int whence) + { + tImageSource* isource = (tImageSource*)fd; + uint64 ret = -1; + do + { + if (whence == SEEK_SET) + { + CC_BREAK_IF(off >= (uint64)isource->size); + ret = isource->offset = (uint32)off; + } + else if (whence == SEEK_CUR) + { + CC_BREAK_IF(isource->offset + off >= (uint64)isource->size); + ret = isource->offset += (uint32)off; + } + else if (whence == SEEK_END) + { + CC_BREAK_IF(off >= (uint64)isource->size); + ret = isource->offset = (uint32)(isource->size-1 - off); + } + else + { + CC_BREAK_IF(off >= (uint64)isource->size); + ret = isource->offset = (uint32)off; + } + } while (0); + + return ret; + } + + static uint64 tiffSizeProc(thandle_t fd) + { + tImageSource* imageSrc = (tImageSource*)fd; + return imageSrc->size; + } + + static int tiffCloseProc(thandle_t fd) + { + CC_UNUSED_PARAM(fd); + return 0; + } + + static int tiffMapProc(thandle_t fd, void** base, toff_t* size) + { + CC_UNUSED_PARAM(fd); + CC_UNUSED_PARAM(base); + CC_UNUSED_PARAM(size); + return 0; + } + + static void tiffUnmapProc(thandle_t fd, void* base, toff_t size) + { + CC_UNUSED_PARAM(fd); + CC_UNUSED_PARAM(base); + CC_UNUSED_PARAM(size); + } +} +#endif // CC_USE_TIFF + bool Image::initWithTiffData(const unsigned char * data, ssize_t dataLen) { - _hasPremultipliedAlpha = true; - - TIFFModule *tiffModule = static_cast(ModuleManager::getModule("tiff")); - if (tiffModule) +#if CC_USE_TIFF + bool ret = false; + do { - DataFromModule dataFromTIFFInit; - bool result = tiffModule->initWithTIFFData(data, dataLen, dataFromTIFFInit); - _width = dataFromTIFFInit.width; - _height = dataFromTIFFInit.height; - _renderFormat = dataFromTIFFInit.renderFormat; - _dataLen = dataFromTIFFInit.dataLength; - _data = dataFromTIFFInit.data; + // set the read call back function + tImageSource imageSource; + imageSource.data = data; + imageSource.size = dataLen; + imageSource.offset = 0; + + TIFF* tif = TIFFClientOpen("file.tif", "r", (thandle_t)&imageSource, + tiffReadProc, tiffWriteProc, + tiffSeekProc, tiffCloseProc, tiffSizeProc, + tiffMapProc, + tiffUnmapProc); + + CC_BREAK_IF(nullptr == tif); + + uint32 w = 0, h = 0; + uint16 bitsPerSample = 0, samplePerPixel = 0, planarConfig = 0; + size_t npixels = 0; - return result; - } - else - { - CCLOG("tiff module is not enabled"); - return false; - } + TIFFGetField(tif, TIFFTAG_IMAGEWIDTH, &w); + TIFFGetField(tif, TIFFTAG_IMAGELENGTH, &h); + TIFFGetField(tif, TIFFTAG_BITSPERSAMPLE, &bitsPerSample); + TIFFGetField(tif, TIFFTAG_SAMPLESPERPIXEL, &samplePerPixel); + TIFFGetField(tif, TIFFTAG_PLANARCONFIG, &planarConfig); + + npixels = w * h; + + _renderFormat = Texture2D::PixelFormat::RGBA8888; + _width = w; + _height = h; + + _dataLen = npixels * sizeof (uint32); + _data = static_cast(malloc(_dataLen * sizeof(unsigned char))); + + uint32* raster = (uint32*) _TIFFmalloc(npixels * sizeof (uint32)); + if (raster != nullptr) + { + if (TIFFReadRGBAImageOriented(tif, w, h, raster, ORIENTATION_TOPLEFT, 0)) + { + /* the raster data is pre-multiplied by the alpha component + after invoking TIFFReadRGBAImageOriented*/ + _hasPremultipliedAlpha = true; + + memcpy(_data, raster, npixels*sizeof (uint32)); + } + + _TIFFfree(raster); + } + + + TIFFClose(tif); + + ret = true; + } while (0); + return ret; +#else + CCLOG("tiff is not enabled, please enalbe it in ccConfig.h"); + return false; +#endif } namespace @@ -1067,7 +1370,7 @@ bool Image::initWithPVRv2Data(const unsigned char * data, ssize_t dataLen) { switch (formatFlags) { case PVR2TexturePixelFormat::PVRTC2BPP_RGBA: - if(!Configuration::getInstance()->supportsPVRTC()) + if (!Configuration::getInstance()->supportsPVRTC()) { CCLOG("cocos2d: Hardware PVR decoder not present. Using software decoder"); _unpack = true; @@ -1081,7 +1384,7 @@ bool Image::initWithPVRv2Data(const unsigned char * data, ssize_t dataLen) heightBlocks = height / 4; break; case PVR2TexturePixelFormat::PVRTC4BPP_RGBA: - if(!Configuration::getInstance()->supportsPVRTC()) + if (!Configuration::getInstance()->supportsPVRTC()) { CCLOG("cocos2d: Hardware PVR decoder not present. Using software decoder"); _unpack = true; @@ -1221,7 +1524,7 @@ bool Image::initWithPVRv3Data(const unsigned char * data, ssize_t dataLen) { case PVR3TexturePixelFormat::PVRTC2BPP_RGB : case PVR3TexturePixelFormat::PVRTC2BPP_RGBA : - if(!Configuration::getInstance()->supportsPVRTC()) + if (!Configuration::getInstance()->supportsPVRTC()) { CCLOG("cocos2d: Hardware PVR decoder not present. Using software decoder"); _unpack = true; @@ -1236,7 +1539,7 @@ bool Image::initWithPVRv3Data(const unsigned char * data, ssize_t dataLen) break; case PVR3TexturePixelFormat::PVRTC4BPP_RGB : case PVR3TexturePixelFormat::PVRTC4BPP_RGBA : - if(!Configuration::getInstance()->supportsPVRTC()) + if (!Configuration::getInstance()->supportsPVRTC()) { CCLOG("cocos2d: Hardware PVR decoder not present. Using software decoder"); _unpack = true; @@ -1250,7 +1553,7 @@ bool Image::initWithPVRv3Data(const unsigned char * data, ssize_t dataLen) heightBlocks = height / 4; break; case PVR3TexturePixelFormat::ETC1: - if(!Configuration::getInstance()->supportsETC()) + if (!Configuration::getInstance()->supportsETC()) { CCLOG("cocos2d: Hardware ETC1 decoder not present. Using software decoder"); int bytePerPixel = 3; @@ -1268,7 +1571,7 @@ bool Image::initWithPVRv3Data(const unsigned char * data, ssize_t dataLen) heightBlocks = height / 4; break; case PVR3TexturePixelFormat::BGRA8888: - if( ! Configuration::getInstance()->supportsBGRA8888()) + if (! Configuration::getInstance()->supportsBGRA8888()) { CCLOG("cocos2d: Image. BGRA8888 not supported on this device"); return false; @@ -1308,7 +1611,7 @@ bool Image::initWithPVRv3Data(const unsigned char * data, ssize_t dataLen) height = MAX(height >> 1, 1); } - if(_unpack) + if (_unpack) { _data = _mipmaps[0].address; _dataLen = _mipmaps[0].len; @@ -1322,7 +1625,7 @@ bool Image::initWithETCData(const unsigned char * data, ssize_t dataLen) const etc1_byte* header = static_cast(data); //check the data - if(!etc1_pkm_is_valid(header)) + if (! etc1_pkm_is_valid(header)) { return false; } @@ -1330,12 +1633,12 @@ bool Image::initWithETCData(const unsigned char * data, ssize_t dataLen) _width = etc1_pkm_get_width(header); _height = etc1_pkm_get_height(header); - if( 0 == _width || 0 == _height ) + if (0 == _width || 0 == _height) { return false; } - if(Configuration::getInstance()->supportsETC()) + if (Configuration::getInstance()->supportsETC()) { //old opengl version has no define for GL_ETC1_RGB8_OES, add macro to make compiler happy. #ifdef GL_ETC1_RGB8_OES @@ -1727,35 +2030,53 @@ bool Image::initWithPVRData(const unsigned char * data, ssize_t dataLen) bool Image::initWithWebpData(const unsigned char * data, ssize_t dataLen) { +#if CC_USE_WEBP + bool ret = false; + #if (CC_TARGET_PLATFORM == CC_PLATFORM_WP8) || (CC_TARGET_PLATFORM == CC_PLATFORM_WINRT) CCLOG("WEBP image format not supported on WinRT or WP8"); - return false; #else - WEBPModule *webpModule = static_cast(ModuleManager::getModule("webp")); - if (webpModule) - { - DataFromModule dataFromWEBPInit; - bool result = webpModule->initWithWEBPData(data, dataLen, dataFromWEBPInit); - _width = dataFromWEBPInit.width; - _height = dataFromWEBPInit.height; - _renderFormat = dataFromWEBPInit.renderFormat; - _dataLen = dataFromWEBPInit.dataLength; - _data = dataFromWEBPInit.data; + do + { + WebPDecoderConfig config; + if (WebPInitDecoderConfig(&config) == 0) break; + if (WebPGetFeatures(static_cast(data), dataLen, &config.input) != VP8_STATUS_OK) break; + if (config.input.width == 0 || config.input.height == 0) break; - return result; - } - else - { - CCLOG("webp module is not enabled"); - return false; - } -#endif + config.output.colorspace = MODE_RGBA; + _renderFormat = Texture2D::PixelFormat::RGBA8888; + _width = config.input.width; + _height = config.input.height; + + _dataLen = _width * _height * 4; + _data = static_cast(malloc(_dataLen * sizeof(unsigned char))); + + config.output.u.RGBA.rgba = static_cast(_data); + config.output.u.RGBA.stride = _width * 4; + config.output.u.RGBA.size = _dataLen; + config.output.is_external_memory = 1; + + if (WebPDecode(static_cast(data), dataLen, &config) != VP8_STATUS_OK) + { + free(_data); + _data = nullptr; + break; + } + + ret = true; + } while (0); +#endif // (CC_TARGET_PLATFORM == CC_PLATFORM_WP8) || (CC_TARGET_PLATFORM == CC_PLATFORM_WINRT) + return ret; +#else + CCLOG("webp is not enabled, please enable it in ccConfig.h"); + return false; +#endif // CC_USE_WEBP } bool Image::initWithRawData(const unsigned char * data, ssize_t dataLen, int width, int height, int bitsPerComponent, bool preMulti) { - bool bRet = false; + bool ret = false; do { CC_BREAK_IF(0 == width || 0 == height); @@ -1772,15 +2093,15 @@ bool Image::initWithRawData(const unsigned char * data, ssize_t dataLen, int wid CC_BREAK_IF(! _data); memcpy(_data, data, _dataLen); - bRet = true; + ret = true; } while (0); - return bRet; + return ret; } #if (CC_TARGET_PLATFORM != CC_PLATFORM_IOS) -bool Image::saveToFile(const std::string& filename, bool bIsToRGB) +bool Image::saveToFile(const std::string& filename, bool isToRGB) { //only support for Texture2D::PixelFormat::RGB888 or Texture2D::PixelFormat::RGBA8888 uncompressed data if (isCompressed() || (_renderFormat != Texture2D::PixelFormat::RGB888 && _renderFormat != Texture2D::PixelFormat::RGBA8888)) @@ -1789,7 +2110,7 @@ bool Image::saveToFile(const std::string& filename, bool bIsToRGB) return false; } - bool bRet = false; + bool ret = false; do { @@ -1804,7 +2125,7 @@ bool Image::saveToFile(const std::string& filename, bool bIsToRGB) if (std::string::npos != strLowerCasePath.find(".png")) { - CC_BREAK_IF(!saveImageToPNG(filename, bIsToRGB)); + CC_BREAK_IF(!saveImageToPNG(filename, isToRGB)); } else if (std::string::npos != strLowerCasePath.find(".jpg")) { @@ -1815,16 +2136,16 @@ bool Image::saveToFile(const std::string& filename, bool bIsToRGB) break; } - bRet = true; + ret = true; } while (0); - return bRet; + return ret; } #endif bool Image::saveImageToPNG(const std::string& filePath, bool isToRGB) { - bool bRet = false; + bool ret = false; do { FILE *fp; @@ -1903,8 +2224,8 @@ bool Image::saveImageToPNG(const std::string& filePath, bool isToRGB) { if (isToRGB) { - unsigned char *pTempData = static_cast(malloc(_width * _height * 3 * sizeof(unsigned char))); - if (nullptr == pTempData) + unsigned char *tempData = static_cast(malloc(_width * _height * 3 * sizeof(unsigned char))); + if (nullptr == tempData) { fclose(fp); png_destroy_write_struct(&png_ptr, &info_ptr); @@ -1918,15 +2239,15 @@ bool Image::saveImageToPNG(const std::string& filePath, bool isToRGB) { for (int j = 0; j < _width; ++j) { - pTempData[(i * _width + j) * 3] = _data[(i * _width + j) * 4]; - pTempData[(i * _width + j) * 3 + 1] = _data[(i * _width + j) * 4 + 1]; - pTempData[(i * _width + j) * 3 + 2] = _data[(i * _width + j) * 4 + 2]; + tempData[(i * _width + j) * 3] = _data[(i * _width + j) * 4]; + tempData[(i * _width + j) * 3 + 1] = _data[(i * _width + j) * 4 + 1]; + tempData[(i * _width + j) * 3 + 2] = _data[(i * _width + j) * 4 + 2]; } } for (int i = 0; i < (int)_height; i++) { - row_pointers[i] = (png_bytep)pTempData + i * _width * 3; + row_pointers[i] = (png_bytep)tempData + i * _width * 3; } png_write_image(png_ptr, row_pointers); @@ -1934,9 +2255,9 @@ bool Image::saveImageToPNG(const std::string& filePath, bool isToRGB) free(row_pointers); row_pointers = nullptr; - if (pTempData != nullptr) + if (tempData != nullptr) { - free(pTempData); + free(tempData); } } else @@ -1962,22 +2283,94 @@ bool Image::saveImageToPNG(const std::string& filePath, bool isToRGB) fclose(fp); - bRet = true; + ret = true; } while (0); - return bRet; + return ret; } bool Image::saveImageToJPG(const std::string& filePath) { - JPEGModule *jpegModule = static_cast(ModuleManager::getModule("jpeg")); - if (jpegModule) +#if CC_USE_JPEG + bool ret = false; + do { - return jpegModule->saveImage(filePath, this); - } - else - { - CCLOG("jpeg module is not enabled"); - return false; - } + struct jpeg_compress_struct cinfo; + struct jpeg_error_mgr jerr; + FILE * outfile; /* target file */ + JSAMPROW row_pointer[1]; /* pointer to JSAMPLE row[s] */ + int row_stride; /* physical row width in image buffer */ + + cinfo.err = jpeg_std_error(&jerr); + /* Now we can initialize the JPEG compression object. */ + jpeg_create_compress(&cinfo); + + CC_BREAK_IF((outfile = fopen(filePath.c_str(), "wb")) == nullptr); + + jpeg_stdio_dest(&cinfo, outfile); + + cinfo.image_width = _width; /* image width and height, in pixels */ + cinfo.image_height = _height; + cinfo.input_components = 3; /* # of color components per pixel */ + cinfo.in_color_space = JCS_RGB; /* colorspace of input image */ + + jpeg_set_defaults(&cinfo); + jpeg_set_quality(&cinfo, 90, TRUE); + + jpeg_start_compress(&cinfo, TRUE); + + row_stride = _width * 3; /* JSAMPLEs per row in image_buffer */ + + if (hasAlpha()) + { + unsigned char *tempData = static_cast(malloc(_width * _height * 3 * sizeof(unsigned char))); + if (nullptr == tempData) + { + jpeg_finish_compress(&cinfo); + jpeg_destroy_compress(&cinfo); + fclose(outfile); + break; + } + + for (int i = 0; i < _height; ++i) + { + for (int j = 0; j < _width; ++j) + + { + tempData[(i * _width + j) * 3] = _data[(i * _width + j) * 4]; + tempData[(i * _width + j) * 3 + 1] = _data[(i * _width + j) * 4 + 1]; + tempData[(i * _width + j) * 3 + 2] = _data[(i * _width + j) * 4 + 2]; + } + } + + while (cinfo.next_scanline < cinfo.image_height) + { + row_pointer[0] = & tempData[cinfo.next_scanline * row_stride]; + (void) jpeg_write_scanlines(&cinfo, row_pointer, 1); + } + + if (tempData != nullptr) + { + free(tempData); + } + } + else + { + while (cinfo.next_scanline < cinfo.image_height) { + row_pointer[0] = & _data[cinfo.next_scanline * row_stride]; + (void) jpeg_write_scanlines(&cinfo, row_pointer, 1); + } + } + + jpeg_finish_compress(&cinfo); + fclose(outfile); + jpeg_destroy_compress(&cinfo); + + ret = true; + } while (0); + return ret; +#else + CCLOG("jpeg is not enabled, please enable it in ccConfig.h"); + return false; +#endif // CC_USE_JPEG } void Image::premultipliedAlpha() @@ -1994,6 +2387,7 @@ void Image::premultipliedAlpha() _hasPremultipliedAlpha = true; } + void Image::setPVRImagesHavePremultipliedAlpha(bool haveAlphaPremultiplied) { _PVRHaveAlphaPremultiplied = haveAlphaPremultiplied; diff --git a/cocos/platform/CCImage.h b/cocos/platform/CCImage.h index bb9e8d0014..483ea686ff 100644 --- a/cocos/platform/CCImage.h +++ b/cocos/platform/CCImage.h @@ -26,8 +26,6 @@ THE SOFTWARE. #ifndef __CC_IMAGE_H__ #define __CC_IMAGE_H__ -#include -#include #include "base/CCRef.h" #include "renderer/CCTexture2D.h" @@ -169,7 +167,6 @@ protected: void premultipliedAlpha(); - protected: /** @brief Determine how many mipmaps can we have. @@ -215,35 +212,6 @@ protected: bool isATITC(const unsigned char *data, ssize_t dataLen); }; -/* Data returned from jpeg/tiff/webp module initialization - */ -typedef struct DataFromModule -{ - Texture2D::PixelFormat renderFormat; - int width; - int height; - ssize_t dataLength; - unsigned char* data; - bool hasPremultiAlpha; -} DataFromModule; - -struct JPEGModule -{ - std::function saveImage; - std::function initWithJPEGData; -}; - -struct TIFFModule -{ - std::function initWithTIFFData; -}; - -struct WEBPModule -{ - std::function initWithWEBPData; -}; - - // end of platform group /// @} diff --git a/docs/RELEASE_NOTES.md b/docs/RELEASE_NOTES.md index 48060de875..78cd8ae2f3 100644 --- a/docs/RELEASE_NOTES.md +++ b/docs/RELEASE_NOTES.md @@ -1,4 +1,4 @@ -# cocos2d-x v3.2 Release Notes # +# cocos2d-x v3.3alpha0 Release Notes # **Table of Contents** *generated with [DocToc](http://doctoc.herokuapp.com/)* @@ -28,9 +28,7 @@ # Misc Information -* Download: http://cdn.cocos2d-x.org/cocos2d-x-3.2.zip -* Full Changelog: https://github.com/cocos2d/cocos2d-x/blob/cocos2d-x-3.2/CHANGELOG -* API Reference: http://www.cocos2d-x.org/reference/native-cpp/V3.2/index.html +* Full Changelog: https://github.com/cocos2d/cocos2d-x/blob/cocos2d-x-3.3alpha0/CHANGELOG * v3.0 Release Notes can be found here: [v3.0 Release Notes](https://github.com/cocos2d/cocos2d-x/blob/cocos2d-x-3.0/docs/RELEASE_NOTES.md) # Requirements @@ -49,7 +47,7 @@ * Xcode 5.1 or newer for iOS or Mac * gcc 4.9 or newer for Linux -* ndk-r9d or newer for Android +* ndk-r9d for Android * Visual Studio 2012 or newer for Windows (win32) * Visual Studio 2012 or newer for Windows Phone 8 @@ -117,179 +115,77 @@ Run Please refer to this document: [ReadMe](../README.md) -# Highlights of v3.2 +# Highlights of v3.3alpha0 -* `Animation3D`/`Animate3D`, new nodes for 3d animation -* `fbx-conv` supports generating binary format which is supported by `Sprite3D` -* Game controller support -* Fast tilemap support -* Added `utils::cpatureScreen` to take screenshot -* Physics body supports scale and rotation -* Added `Node::enumerateChildren` and `utils::findChildren`, and support c++ 11 regular expression -* Added `Node::setNormalizedPosition`, `Node`'s position in pixel will be calculated according its parent's content size - -# Documents - -* [Sprite3D & Animation3D](http://cocos2d-x.org/wiki/Sprite3D) -* [Game controller](http://www.cocos2d-x.org/wiki/Game_Controller) -* [How to compile shader on WP8](http://cocos2d-x.org/wiki/How_to_update_wp8_shader) - -# Toolchain requirement changed - -`Node::enumerateChildren()` uses `std::regex` which will cause crash using gcc v4.8 or lower version. -Because `OTHER_LDFLAGS` can not work in Xcode6 beta3. So we used fat library(including 64-bit libaries) on iOS. But Xcode 5.0 or lower version has building problem by this way. - -So - -* NDK r9d or newer version is required for Android building -* gcc 4.9 is required for linux building -* Xcode 5.1 or newer is required on iOS - -# atof issue on Android - -We found a bug of `atof` on Android when using libc++. The bug is that, the return value of `atof` may be `-inf` when passing some valid digit string. - -For example - -```c++ -atof("90.099998474121094"); // -> return value is -inf -``` - -We have reported it to google guys, and they confirmed that it is a bug. In order to work around this issue, we added `utils::atof()`. - -The corresponding pull request for this issue is [here](https://github.com/cocos2d/cocos2d-x/pull/7440). You can refer to this pull request for demail information. +* 3d: `Camera`, `AABB`, `OBB` and `Ray` +* ui: added `Scale9Sprite` +* FileUitls: added `isDirectoryExist()`, `createDirectory()`, `removeDirectory()`, `removeFile()`, `renameFile()` and `getFileSize()` +* Device: added `setKeepScreenOn()` on iOS and Android +* Added c++11 random support +* RenderTexture: added a call back function for `saveToFile()` +* SpriteFrameCache: support loading from plist file content data +* Many other small features added and many bugs fixed # Features in detail -## Sprite3D & Animation3D +## Camera + +This version of camera is powerful then previous one. And you can add it as a child anywhere. If you want to let a Node to be visited by a camera, Node's camera mask should include Camera's flag: -Sample code to use binary version ```c++ -auto sprite3d = Sprite3D::create("filename.c3b"); -addChild(sprite3d); - -auto animation3d = Animation3D("filename.c3b"); -auto animate3d = Animate3D::create(animation3d); -sprite3d->runAction(RepeatForEver::create(animate)); +// let sprite to be visited by a camera +auto sprite = Sprite::create("myFile.png"); +sprite->setCameraMask(CameraFlag::USER1); +auto camera = Camera::createPerspective(60, winSize.width/winSize.height, 1, 1000); +camera->setCameraFlag(CameraFlag::USER1); +scene->addChild(camera); ``` -Detail information please refer to [Sprite3D & Animation3D](http://cocos2d-x.org/wiki/Sprite3D). +If you have many Nodes that want to be visited by a camera, there is a convenient way: -### `fbx-conv` usage - -* Mac OS X - -``` -$ cd COCOS2DX_ROOT/tools/fbx-conv/mac -$ ./fbx-conv [-a|-b|-t] FBXFile -``` - -* Windows - -``` -cd COCOS2DX_ROOT/tools/fbx-conv/windows -fbx-conv [-a|-b|-t] FBXFile -``` - -Options: - -* -a: export both text and binary format -* -b: export binary format -* -t: export text format - -## Game controller - -Supported controller type: - -* Android standard controllers -* Amazon tv -* OUYA -* Moga -* Nibiru -* iOS standard controllers - -Sample codes ```c++ -// register event listener -auto listener = EventListenerController::create(); -listner->onKeyDown = ... +auto layer = Layer::create(); +auto sprite1 = Sprite::create(); +auto sprite2 = Sprite::create(); +layer->addChild(sprite1); +layer->addChild(sprite2); +// it will set camera mask for all its children +layer->setCameraMask(CameraFlg::USER1); + +auto camera = Camera::createPerspective(); +camera->setCameraFlag(CameraFlag::USER1); +scene->addChild(camera); +``` + +Full test case please refer to `tests/cpp-tests/res/Camera3DTest/Camera3DTest.cpp`. + +## AABB, OBB and Ray + +TBD + +## ui::Scale9Sprite + +TBD + +## c++11 random support + +Since `rand()` is not good(refer to [this document](http://c-faq.com/lib/randrange.html)), we use c++11 random library to do generate random number, and provide a function to easily using: + +```c++ +int randInt = cocos2d::random(1, 10); +float randFloat = cocos2d::random(1.f, 10.f); +``` + +## RenderTexture save function + +`RenderTexture::saveToFile()` will not save rendertexture when the function returns, because it just send render command to renderer. The file will be saved after render command is executed. It is not convenient if you want to use the saved file to do some work. So we added a parameter in `RenderTexture::saveToFile()` to set a call back function when the file is saved. + +```c++ +renderTexture->begin(); ... -eventDispatcher->addEventListenerWithSceneGraphPriority(listener, this); +renderTexture->end(); -// start connecting controller -Controller::startDiscoveryController(); +renderTexture->saveToFile("myFile.png", true, callback); -// handler key down/ key up event -void GameControllerTest::onKeyDown(Controller *controller, int keyCode, Event *event) -{ - switch (keyCode) - { - case Controller::Key::BUTTON_A: - ... - break; - - ... - } -} ``` -Detail information please refer to [Game controller](http://www.cocos2d-x.org/wiki/Game_Controller). - -## Fast tilemap - -Fast tilemap has the same API as `TMXTiledMap` without deprecated functions. - -Sample code -```c++ -auto tilemap = cocos2d::experimental::TMXTiledMap::create("MyFile.tmx"); -addChild(tilemap); -``` - -Full demo please refer to `COCOS2DX_ROOT/tests/cpp-tests/Classes/TileMapTest/TileMapTest2.cpp`. - -## Node::enumerateChildren - -This functions is used to enumerate children of a `Node`. It supports c++ 11 regular expression. - -```c++ -// Find nodes whose name is 'nameToFind' and end with digits. -node->enumerateChildren("nameToFind[[:digit:]]+", [](Node* node) -> bool { - ... - return false; // return true to stop at first match -}); - -// Find nodes whose name is 'nameToFind' and end with digits recursively. -node->enumerateChildren("nameToFind[[:digit:]]+", [](Node* node) -> bool { - ... - return false; // return true to stop at first match -}); -``` - -Full test please refer to `NodeNameTest` in `COCOS2DX_ROOT/tests/cpp-tests/NodeTest/NodeTest.cpp`. - -Because this function uses `std::regex` which is not supported well in gcc 4.8 or lower version. So we use `clang` and `stdc++` instead for Android building. This lead to the result that `NDK r9d` or newer is required. And `gcc 4.9` is required on linux. - -## utils::findChildren - -This is a helper function to find children of a `Node` share a name. The implementation of this function bases on `Node::enumerateChildren`. - -```c++ -auto children = utils::findChildren(node, "nameToFind"); - -... -``` - -## Node::setNormalizedPosition - -Can use this function to set Node's position(x,y) using value between 0 and 1. `Can use this function when it has a parent node.` The positions in pixels is calculated like the following: - -```c++ -// pseudo code -void setNormalizedPosition(Vec2 pos) -{ - Size s = getParent()->getContentSize(); - _position = pos * s; -} -``` - -Full test please refer to `NodeNormalizedPositionTest1/2` in `tests/cpp-tests/Classes/NodeTest/NodeTest.cpp`. diff --git a/templates/cpp-template-default/cocos-project-template.json b/templates/cpp-template-default/cocos-project-template.json index 9d4ee7d291..6002d95fd4 100644 --- a/templates/cpp-template-default/cocos-project-template.json +++ b/templates/cpp-template-default/cocos-project-template.json @@ -27,6 +27,7 @@ "proj.win32/PROJECT_NAME.sln", "proj.win32/main.cpp", "proj.wp8-xaml/App/PROJECT_NAME.csproj", + "proj.wp8-xaml/App/Properties/WMAppManifest.xml", "proj.wp8-xaml/PROJECT_NAME.sln", "proj.android/.project", "proj.android/.cproject", diff --git a/templates/cpp-template-default/proj.wp8-xaml/App/Properties/WMAppManifest.xml b/templates/cpp-template-default/proj.wp8-xaml/App/Properties/WMAppManifest.xml index 8437827d62..8622bf45fc 100644 --- a/templates/cpp-template-default/proj.wp8-xaml/App/Properties/WMAppManifest.xml +++ b/templates/cpp-template-default/proj.wp8-xaml/App/Properties/WMAppManifest.xml @@ -1,7 +1,7 @@  - + Assets\ApplicationIcon.png diff --git a/tools/cocos2d-console b/tools/cocos2d-console index 802e261172..fb44ec8cf7 160000 --- a/tools/cocos2d-console +++ b/tools/cocos2d-console @@ -1 +1 @@ -Subproject commit 802e261172e2f3b49d1feed20935a53820694ab2 +Subproject commit fb44ec8cf77f6f51503987c3a1a0964580b5d88f