diff --git a/README.md b/README.md index cd4feb8..56e4d3a 100644 --- a/README.md +++ b/README.md @@ -1,50 +1,195 @@ -# Welcome to your Expo app 👋 +# react-native-linear-gradient -This is an [Expo](https://expo.dev) project created with [`create-expo-app`](https://www.npmjs.com/package/create-expo-app). +A `` element for React Native -## Get started +[![ci][1]][2] +[![npm version][3]][4] +[![npm downloads][5]][4] -1. Install dependencies +

+ +

- ```bash - npm install - ``` +## Table of Contents -2. Start the app +- [Installation](#installation) +- [Usage and Examples](#examples) +- [Props](#props) +- [Example App](#an-example-app) +- [Troubleshooting](#troubleshooting) +- [Other Platforms](#other-platforms) - ```bash - npx expo start - ``` +## Installation -In the output, you'll find options to open the app in a - -- [development build](https://docs.expo.dev/develop/development-builds/introduction/) -- [Android emulator](https://docs.expo.dev/workflow/android-studio-emulator/) -- [iOS simulator](https://docs.expo.dev/workflow/ios-simulator/) -- [Expo Go](https://expo.dev/go), a limited sandbox for trying out app development with Expo - -You can start developing by editing the files inside the **app** directory. This project uses [file-based routing](https://docs.expo.dev/router/introduction). - -## Get a fresh project - -When you're ready, run: - -```bash -npm run reset-project +```sh +yarn add react-native-linear-gradient ``` -This command will move the starter code to the **app-example** directory and create a blank **app** directory where you can start developing. +Or, using npm: `npm install react-native-linear-gradient` -## Learn more +## Examples -To learn more about developing your project with Expo, look at the following resources: +[react-native-login](https://github.com/brentvatne/react-native-login) is a +legacy component which showcases the use of ``. -- [Expo documentation](https://docs.expo.dev/): Learn fundamentals, or go into advanced topics with our [guides](https://docs.expo.dev/guides). -- [Learn Expo tutorial](https://docs.expo.dev/tutorial/introduction/): Follow a step-by-step tutorial where you'll create a project that runs on Android, iOS, and the web. +### Simple -## Join the community +The following code will produce something like this: -Join our community of developers creating universal apps. +![Example code result](https://raw.githubusercontent.com/react-native-community/react-native-linear-gradient/HEAD/images/example.png) -- [Expo on GitHub](https://github.com/expo/expo): View our open source platform and contribute. -- [Discord community](https://chat.expo.dev): Chat with Expo users and ask questions. +```javascript +import LinearGradient from 'react-native-linear-gradient'; + +// Within your render function + + + Sign in with Facebook + + + +// Later on in your styles.. +var styles = StyleSheet.create({ + linearGradient: { + flex: 1, + paddingLeft: 15, + paddingRight: 15, + borderRadius: 5 + }, + buttonText: { + fontSize: 18, + fontFamily: 'Gill Sans', + textAlign: 'center', + margin: 10, + color: '#ffffff', + backgroundColor: 'transparent', + }, +}); +``` + +### Horizontal gradient + +Using the styles from above, set `start` and `end` like this to make the gradient go from left to right, instead of from top to bottom: + +```javascript + + + Sign in with Facebook + + +``` + +### Text gradient (iOS) + +On iOS you can use the `MaskedViewIOS` to display text with a gradient. The trick here is to render the text twice; once for the mask, and once to let the gradient have the correct size (hence the `opacity: 0`): + +```jsx +}> + + + + +``` + +### Animated Gradient + +Check out the [example app](https://github.com/react-native-linear-gradient/react-native-linear-gradient/tree/HEAD/example/) (`git clone` this project, cd into it, npm install, open in Xcode and run) to see how this is done: + +![Example with extra props](https://raw.githubusercontent.com/react-native-community/react-native-linear-gradient/HEAD/images/example-animated.gif) + +*This gif was created using [licecap](http://www.cockos.com/licecap/) - a great piece of free OSS* + +### Transparent Gradient + +The use of `transparent` color will most likely not lead to the expected result. `transparent` is actually a transparent black color (`rgba(0, 0, 0, 0)`). If you need a gradient in which the color is "fading", you need to have the same color with changing alpha channel. Example: + +```jsx +// RGBA + + + +// Hex + + +``` + +## Props + +In addition to regular `View` props, you can also provide additional props to customize your gradient look: + +#### colors + +An array of at least two color values that represent gradient colors. Example: `['red', 'blue']` sets gradient from red to blue. + +#### start + +An optional object of the following type: `{ x: number, y: number }`. Coordinates declare the position that the gradient starts at, as a fraction of the overall size of the gradient, starting from the top left corner. Example: `{ x: 0.1, y: 0.1 }` means that the gradient will start 10% from the top and 10% from the left. + +#### end + +Same as start, but for the end of the gradient. + +#### locations + +An optional array of numbers defining the location of each gradient color stop, mapping to the color with the same index in `colors` prop. Example: `[0.1, 0.75, 1]` means that first color will take 0% - 10%, second color will take 10% - 75% and finally third color will occupy 75% - 100%. + +```javascript + + + Sign in with Facebook + + +``` + +![Example with extra props](https://raw.githubusercontent.com/react-native-community/react-native-linear-gradient/HEAD/images/example-other-props.png) + +#### useAngle / angle / angleCenter + +You may want to achieve an angled gradient effect, similar to those in image editors like Photoshop. +One issue is that you have to calculate the angle based on the view's size, which only happens asynchronously and will cause unwanted flickr. + +In order to do that correctly you can set `useAngle={true} angle={45} angleCenter={{x:0.5,y:0.5}}`, to achieve a gradient with a 45 degrees angle, with its center positioned in the view's exact center. + +`useAngle` is used to turn on/off angle based calculation (as opposed to `start`/`end`). +`angle` is the angle in degrees. +`angleCenter` is the center point of the angle (will control the weight and stretch of the gradient like it does in photoshop. + +## An example app + +You can see this component in action in [brentvatne/react-native-login](https://github.com/brentvatne/react-native-login/blob/HEAD/App/Screens/LoginScreen.js#L58-L62). + +## Troubleshooting + +### iOS build fails: library not found, "BVLinearGradient" was not found in the UIManager + +1. Ensure to run `pod install` before running the app on iOS +2. Ensure you use `ios/**.xcworkspace` file instead of `ios./**.xcodeproj` + +### Other + +Clearing build caches and reinstalling dependencies sometimes solve some issues. Try next steps: + +1. Reinstalling `node_modules` with `rm -rf node_modules && yarn` +2. Clearing Android Gradle cache with `(cd android && ./gradlew clean)` +3. Reinstalling iOS CocoaPods with `(cd ios && rm -rf ./ios/Pods/**) && npx pod-install` +4. Clearing Xcode Build cache (open Xcode and go to Product -> Clean Build Folder) + +For other troubleshooting issues, go to [React Native Troubleshooting](https://reactnative.dev/docs/troubleshooting.html) + +## Other platforms + +- Web: [react-native-web-community/react-native-web-linear-gradient](https://github.com/react-native-web-community/react-native-web-linear-gradient) + +## License + +MIT + +[1]: https://github.com/react-native-linear-gradient/react-native-linear-gradient/workflows/ci/badge.svg +[2]: https://github.com/react-native-linear-gradient/react-native-linear-gradient/actions +[3]: https://img.shields.io/npm/v/react-native-linear-gradient.svg +[4]: https://www.npmjs.com/package/react-native-linear-gradient +[5]: https://img.shields.io/npm/dm/react-native-linear-gradient.svg diff --git a/README.react-native-linear-gradient.md b/README.react-native-linear-gradient.md new file mode 100644 index 0000000..56e4d3a --- /dev/null +++ b/README.react-native-linear-gradient.md @@ -0,0 +1,195 @@ +# react-native-linear-gradient + +A `` element for React Native + +[![ci][1]][2] +[![npm version][3]][4] +[![npm downloads][5]][4] + +

+ +

+ +## Table of Contents + +- [Installation](#installation) +- [Usage and Examples](#examples) +- [Props](#props) +- [Example App](#an-example-app) +- [Troubleshooting](#troubleshooting) +- [Other Platforms](#other-platforms) + +## Installation + +```sh +yarn add react-native-linear-gradient +``` + +Or, using npm: `npm install react-native-linear-gradient` + +## Examples + +[react-native-login](https://github.com/brentvatne/react-native-login) is a +legacy component which showcases the use of ``. + +### Simple + +The following code will produce something like this: + +![Example code result](https://raw.githubusercontent.com/react-native-community/react-native-linear-gradient/HEAD/images/example.png) + +```javascript +import LinearGradient from 'react-native-linear-gradient'; + +// Within your render function + + + Sign in with Facebook + + + +// Later on in your styles.. +var styles = StyleSheet.create({ + linearGradient: { + flex: 1, + paddingLeft: 15, + paddingRight: 15, + borderRadius: 5 + }, + buttonText: { + fontSize: 18, + fontFamily: 'Gill Sans', + textAlign: 'center', + margin: 10, + color: '#ffffff', + backgroundColor: 'transparent', + }, +}); +``` + +### Horizontal gradient + +Using the styles from above, set `start` and `end` like this to make the gradient go from left to right, instead of from top to bottom: + +```javascript + + + Sign in with Facebook + + +``` + +### Text gradient (iOS) + +On iOS you can use the `MaskedViewIOS` to display text with a gradient. The trick here is to render the text twice; once for the mask, and once to let the gradient have the correct size (hence the `opacity: 0`): + +```jsx +}> + + + + +``` + +### Animated Gradient + +Check out the [example app](https://github.com/react-native-linear-gradient/react-native-linear-gradient/tree/HEAD/example/) (`git clone` this project, cd into it, npm install, open in Xcode and run) to see how this is done: + +![Example with extra props](https://raw.githubusercontent.com/react-native-community/react-native-linear-gradient/HEAD/images/example-animated.gif) + +*This gif was created using [licecap](http://www.cockos.com/licecap/) - a great piece of free OSS* + +### Transparent Gradient + +The use of `transparent` color will most likely not lead to the expected result. `transparent` is actually a transparent black color (`rgba(0, 0, 0, 0)`). If you need a gradient in which the color is "fading", you need to have the same color with changing alpha channel. Example: + +```jsx +// RGBA + + + +// Hex + + +``` + +## Props + +In addition to regular `View` props, you can also provide additional props to customize your gradient look: + +#### colors + +An array of at least two color values that represent gradient colors. Example: `['red', 'blue']` sets gradient from red to blue. + +#### start + +An optional object of the following type: `{ x: number, y: number }`. Coordinates declare the position that the gradient starts at, as a fraction of the overall size of the gradient, starting from the top left corner. Example: `{ x: 0.1, y: 0.1 }` means that the gradient will start 10% from the top and 10% from the left. + +#### end + +Same as start, but for the end of the gradient. + +#### locations + +An optional array of numbers defining the location of each gradient color stop, mapping to the color with the same index in `colors` prop. Example: `[0.1, 0.75, 1]` means that first color will take 0% - 10%, second color will take 10% - 75% and finally third color will occupy 75% - 100%. + +```javascript + + + Sign in with Facebook + + +``` + +![Example with extra props](https://raw.githubusercontent.com/react-native-community/react-native-linear-gradient/HEAD/images/example-other-props.png) + +#### useAngle / angle / angleCenter + +You may want to achieve an angled gradient effect, similar to those in image editors like Photoshop. +One issue is that you have to calculate the angle based on the view's size, which only happens asynchronously and will cause unwanted flickr. + +In order to do that correctly you can set `useAngle={true} angle={45} angleCenter={{x:0.5,y:0.5}}`, to achieve a gradient with a 45 degrees angle, with its center positioned in the view's exact center. + +`useAngle` is used to turn on/off angle based calculation (as opposed to `start`/`end`). +`angle` is the angle in degrees. +`angleCenter` is the center point of the angle (will control the weight and stretch of the gradient like it does in photoshop. + +## An example app + +You can see this component in action in [brentvatne/react-native-login](https://github.com/brentvatne/react-native-login/blob/HEAD/App/Screens/LoginScreen.js#L58-L62). + +## Troubleshooting + +### iOS build fails: library not found, "BVLinearGradient" was not found in the UIManager + +1. Ensure to run `pod install` before running the app on iOS +2. Ensure you use `ios/**.xcworkspace` file instead of `ios./**.xcodeproj` + +### Other + +Clearing build caches and reinstalling dependencies sometimes solve some issues. Try next steps: + +1. Reinstalling `node_modules` with `rm -rf node_modules && yarn` +2. Clearing Android Gradle cache with `(cd android && ./gradlew clean)` +3. Reinstalling iOS CocoaPods with `(cd ios && rm -rf ./ios/Pods/**) && npx pod-install` +4. Clearing Xcode Build cache (open Xcode and go to Product -> Clean Build Folder) + +For other troubleshooting issues, go to [React Native Troubleshooting](https://reactnative.dev/docs/troubleshooting.html) + +## Other platforms + +- Web: [react-native-web-community/react-native-web-linear-gradient](https://github.com/react-native-web-community/react-native-web-linear-gradient) + +## License + +MIT + +[1]: https://github.com/react-native-linear-gradient/react-native-linear-gradient/workflows/ci/badge.svg +[2]: https://github.com/react-native-linear-gradient/react-native-linear-gradient/actions +[3]: https://img.shields.io/npm/v/react-native-linear-gradient.svg +[4]: https://www.npmjs.com/package/react-native-linear-gradient +[5]: https://img.shields.io/npm/dm/react-native-linear-gradient.svg diff --git a/components/ui/Message.tsx b/README.whisperrn.md similarity index 100% rename from components/ui/Message.tsx rename to README.whisperrn.md diff --git a/__mocks__/jestSetup.ts b/__mocks__/jestSetup.ts new file mode 100644 index 0000000..bb23898 --- /dev/null +++ b/__mocks__/jestSetup.ts @@ -0,0 +1 @@ +jest.mock('react-native/Libraries/Animated/NativeAnimatedHelper'); \ No newline at end of file diff --git a/app/(tabs)/index.tsx b/app/(tabs)/index.tsx index 291255a..bdd0588 100644 --- a/app/(tabs)/index.tsx +++ b/app/(tabs)/index.tsx @@ -4,14 +4,24 @@ import { Text } from 'react-native'; import { Image, StyleSheet, Platform } from 'react-native'; import { Conversation, Speaker } from '../lib/conversation'; import { language_matrix_entry, Translator } from '../i18n/api'; +import ConversationThread from '@/components/ConversationThread'; export default function HomeScreen() { const [language, setLanguage] = useState() + const [conversation, setConversation] = useState(); return ( - language ? : - setLanguage(l)} /> + conversation ? : + { + setConversation(new Conversation( + new Translator("en", language.code), + {id: "host", language: "en"}, + {id: "guest", language: language.code}, + )) + } + } /> ); } diff --git a/app/lib/__test__/conversation.test.tsx b/app/lib/__test__/conversation.test.tsx index 1a5046f..e3813a4 100644 --- a/app/lib/__test__/conversation.test.tsx +++ b/app/lib/__test__/conversation.test.tsx @@ -9,21 +9,21 @@ describe('Conversation', () => { beforeEach(() => { const translator = new Translator("en", "es"); - const s1: Speaker = { language: "en", id: "s1" }; - const s2: Speaker = { id: "s2", language: "es" } + const s1: Speaker = { language: "en", id: "host" }; + const s2: Speaker = { id: "guest", language: "es" } conversation = new Conversation(translator, s1, s2); }); it('should add a message to the conversation', () => { - conversation.addMessage({ id: "s1", language: "en" }, "Hello"); + conversation.addMessage({ id: "host", language: "en" }, "Hello"); expect(conversation.length).toBe(1); - expect(conversation[0].speaker.id).toEqual("s1"); + expect(conversation[0].speaker.id).toEqual("host"); expect(conversation[0].text).toEqual("Hello"); }); it('should translate the last message in the conversation', async () => { - conversation.addMessage({ id: "s1", language: "en" }, "Hello"); + conversation.addMessage({ id: "host", language: "en" }, "Hello"); await conversation.translateLast(); expect(conversation[conversation.length - 1].translation).toEqual("Hola"); diff --git a/app/lib/conversation.ts b/app/lib/conversation.ts index 6008271..26be6cb 100644 --- a/app/lib/conversation.ts +++ b/app/lib/conversation.ts @@ -5,33 +5,54 @@ export type Speaker = { language : string, } + export class Message { public translation? : string; - constructor (public text : string, public speaker : Speaker) {} + public onTextUpdate?: (message: Message) => void; + public onTextDone?: (message: Message) => Promise; + public onTranslationDone?: (message: Message) => void; + + constructor (public conversation : Conversation, public speaker : Speaker, public text? : string) {} public async translate(translator : Translator, language? : string) { + if (!this.text) throw new Error("No text") this.translation = await translator.translate(this.text, language); } + + get otherSpeaker() { + return this.speaker.id === this.conversation.host.id ? this.conversation.guest : this.conversation.host + } + + get otherLanguage() { + return this.otherSpeaker.language + } } export class Conversation extends Array { + + public onAddMessage? : (conversation : Conversation) => Promise; + public onTranslationDone? : (conversation : Conversation) => Promise; + constructor ( private translator : Translator, - public s1 : Speaker, - public s2 : Speaker, + public host : Speaker, + public guest : Speaker, ) { super(); } - public addMessage(speaker : Speaker, text : string) { - this.push(new Message(text, speaker)); + public addMessage(speaker : Speaker, text? : string) { + this.push(new Message(this, speaker, text)); } public async translateMessage(i : number) { if (!this[i]) throw new Error(`${i} is not a valid message number`); - const otherSpeaker = this[i].speaker.id === this.s1.id ? this.s2 : this.s1 - console.log(`Translating sentence to %s: %s`, otherSpeaker.language, this[i].text) - await this[i].translate(this.translator, otherSpeaker.language); + console.log(`Translating sentence to %s: %s`, this[i].otherLanguage, this[i].text) + await this[i].translate(this.translator, this[i].otherLanguage); + } + + get lastMessage() { + return this[this.length-1] } public async translateLast() { diff --git a/babel.config.js b/babel.config.js index a50f080..9baf3fd 100644 --- a/babel.config.js +++ b/babel.config.js @@ -1,6 +1,8 @@ -module.exports = { - presets: [ - ['@babel/preset-env', {targets: {node: 'current'}}], - '@babel/preset-typescript', - ], - }; \ No newline at end of file +// module.exports = { +// presets: [ +// ['@babel/preset-env', {targets: {node: 'current'}}], +// '@babel/preset-typescript', +// ], +// }; + +export const presets = ['module:metro-react-native-babel-preset', 'module:@babel/plugin-transform-private-methods',]; \ No newline at end of file diff --git a/components/ConversationPlay.tsx b/components/ConversationPlay.tsx deleted file mode 100644 index dcb3b75..0000000 --- a/components/ConversationPlay.tsx +++ /dev/null @@ -1,107 +0,0 @@ - -interface ConversationPlayProps { - translator: Translator; -} - -function ConversationPlay (props : ConversationPlayProps) { - const [language, setLanguage] = useState(languages[0].code); - const [message, setMessage] = useState(''); - const [conversation, setConversation] = useState(null); - - const startConversation = () => { - const speaker: Speaker = { id: "user", language }; - const newConversation = new Conversation(translator, speaker); - setConversation(newConversation); - }; - - const addMessage = async () => { - if (conversation) { - conversation.addMessage({ id: "user", language }, message); - await conversation.translateLast(); - setMessage(''); - } - }; - - return ( - - Select Language: - {languages.map(lang => ( -